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前 有 言 


Qreword 


Android 是 一 个 以 Linux 为 基础 的 半 开 源 操作 系统 ,主要 用 于 管理 手机 、 手 表 、 眼 镜 、 电 视 等 智能 
设备 。Android 操作 系统 最 初 由 Andy Rubin 开发 ,2005 年 8 月 由 Google 收购 注资 。2007 年 11 月， 
Google 与 84 家 人 硬件 制造 商 、 软 件 开发 商 及 电信 营运 商 组 建 开 放手 机 联盟 ,共同 研发 改进 Android 操 
作 系 统 。 第 一 部 Android 智能 手机 发 布 于 2008 年 10 月 ,在 随后 几 年 中 ,Android 开始 了 迅猛 发 展 的 
历程 ,并 成 为 全 球 最 主要 的 移动 端 操作 系统 。 

随 着 支持 Android 操作 系统 的 智能 设备 的 不 断 普 及 和 推广 ,运行 在 Android 操作 系统 上 的 智能 
应 用 项 目 也 如 雨后春笋 般 涌现 。 从 IT 发 展 潮流 来 看 , 越 来 越 丰 富 的 移动 应 用 是 大 势 所 趋 ,手机 支付 、 
手机 拍照 .手机 游戏 .手机 导航 、 物 联网 等 不 断 改变 着 人 们 的 生活 方式 和 工作 方式 。 一 种 优秀 的 
Android 应 用 即 可 造就 一 家 IT 公司 .打造 一 个 产业 链 、 诞 生 十 个 富豪 ,这 已 不 再 是 神话 。 如 何 将 最 新 
的 技术 、 理 念 和 创意 融入 到 应 用 开发 中 ,是 每 个 Android 程序 员 需 要 不 断 思 考 的 问题 ,也 是 本 书 创作 
的 初衷。 

本 书 以 “问题 描述 十 解决 方案 ”的 模式 ,以 Android 5. 0 为 核心 例 举 了 300 个 实用 性 极 强 的 移动 端 
应 用 开发 案例 , 旨 在 帮助 广大 读者 快速 解决 实际 开发 过 程 中 面临 的 诸多 问题 ,从 而 不 断 提 高 开发 效 
率 .拓展 应 用 领域 。 全 书 根 据 实例 功能 将 内 容 分 为 UI 布局 .常用 控件 .文字 .图形 和 图 像 动画、 音频 
和 视频 文件 和 数据 .系统 和 设备 .第 三 方 SDK 开发 等 9 章 , 以 所 见 即 所 得 .所 学 即 所 用 的 速成 思维 方 
式 展示 了 个 性 化 布局 .特效 文字 和 图 形 、 矢 量 图形 动态 绘制 .颜色 和 矩阵 特效 .PorterDuff 特效 、 路 径 特 
效 、 场 景 过 渡 ,三维 空 间 旋转 、 图 像 轮 播 .头像 裁剪 、 网 格 图 像 动画 .旋转 3D 地 球 、 各 种 传感器 应 用 、 网 
络 文件 断 点 续 传 等 超 炫 超 酷 实例 的 实现 过 程 和 代码 。 使 用 第 三 方 SDK 进行 应 用 开发 ,如 使 用 腾讯 
SDK 实现 将 文本 .图像 .音乐 .视频 等 分 享 到 QQ 好 友 及 微 信 朋 友 圈 等 ; 使 用 百度 SDK 实现 查询 驾车 
和 步行 线路 、 查 询 指 定 城市 的 热力 图 、 调 用 百度 地 图 的 导航 功能 、 以 俯视 角度 观察 街道 三 维 图 、 在 百度 
地 图 上 添加 悬浮 框 和 动画 等 ; 使 用 新 浪 SDK 实现 将 图 像 发 布 到 微 博 、 根 据 微 博 简 介 内 容 生 成 二 维 
码 等 。 

本 书 所 有 实例 均 基于 Android 5.0, 在 Android Studio 2. 2 集成 开发 环境 中 使 用 Java 和 XML 语 
言 编写 ,因此 测试 手机 或 模拟 器 的 Android 版 本 不 能 低 于 5. 0。 部 分 实例 在 模拟 器 上 无 法 测试 ,建议 
在 学 习 时 使 用 屏幕 分 辨 率 为 1920X1080 像素 ,操作 系统 为 Android 5.0 及 其 以 上 版 本 的 手机 作为 主 
要 测试 工具 。 

全 书 所 有 内 容 和 思想 并 非 一 人 之 力所能及 ,而 是 凝聚 了 众多 热心 人 士 的 智慧 并 经 过 充分 的 提炼 
和 总 结 而 成 ,在 此 对 他 们 表示 崇高 的 敬意 和 衷心 的 感谢 ! 本 书 编写 人 员 包 括 罗 帅 、 罗 斌 、 汪 明 云 、 曹 
勇 、 陈 宁 、 邓 承 惠 . 邓 小 渝 . 范 刚 强 、 何 守 怕 洪亮 . 洪 沛 林 、 江 素 芳 . 蓝 洋 、 雷 国 忠 、 雷 惠 、 雷 玲 、 雷 平 、 雷 治 
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英 、 刘 恭 德 、, 刘 兴 红 、 罗 职 、 唐 静 、 唐 兴 忠 、. 童 结 嘉 \、 汪 兰 、 王 彬 、 王 伯 芳 、 王 年 素 、 王 正 建 . 吴 多 、 吴 诗 华 、 杨 
开平 、 杨 蕉 、. 易 伶 . 张 志 红 、 郑 少 文 等 ,书稿 由 罗 斌 完成 统 稿 。 由 于 时 间 关 系 和 作者 水 平原 因 , 少 量 内 容 
可 能 存在 认识 不 全 面 或 有 偶 颇 的 问题 ,以 及 一 些 踊 漏 和 不 当 之 处 , 敬 请 读者 批评 指正 。 

读者 可 将 购书 凭证 发 送 至 邮箱 huangzh@tup. tsinghua. edu. cn, 索 取 本 书 源 代码 。 


罗 帅 ” 罗 试 
2019 年 于 重庆 渝 北 
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030 


使 用 纯 Java 代码 创建 应 用 UI 界面 


使 用 自 定 义 View 代替 布局 文件 … 


有 TableLasont 市 局 二 本 时兴 en 


使 用 TextInputLayonut 管理 输入 框 提 示 


使 用 GridLayout 创建 计算 器 按键 布局 
使 用 RelativeLayout 按照 相 邻 关系 布局 pp 
使 用 ConstraintLayout 在 右 下 角 布局 
使 用 TableLayout 拉 伸 控件 填充 容器 .pp 
使 用 TableLayout 缩小 控件 适应 容器 pp 
使 用 LinearLayout 纵 癌 居中 对 齐 控件 pp 
使 用 LinearLayout 按 权 重 分配 探 件 室 间 .PP 


使 用 ConstraintLayout 平分 剩余 空间 … 


使 用 ConstraintLayout 无 间隙 布局 控件 PP 
使 用 TabLayout 和 适配器 创建 选项 卡 .pp 
使 用 TabLayout 和 Fragment 创建 选项 卡 pp 
使 用 FrameLayout 创建 纵 加 选项 卡 pp 


使 用 TabHost 创建 横向 选项 卡 
使 用 AbsoluteLayout 实现 平移 控件 


使 用 FrameLayout 实现 闪烁 控件 .ee 
自 定 文 FrameLayout 创建 得 页 卷 边 动 丽 。 soso 
,TO OO PT 


在 ‘TextView 中 创建 空 s /> ` 文 字 了 
在 TextView 中 实现 上 文 下 图 的 布局 pp 
在 TextView 中 为 文本 添加 超 链 接 pp 


在 自 定 义 View 中 实现 垂直 滚动 文本 . 
在 EditText 中 指定 输入 法 的 数字 软 键盘 
禁止 在 EditText 中 插入 非 字 符 表情 符号 


使 用 AutoCompleteTextView 实现 自动 提示 pp 
使 用 SearchView 和 ListView 实现 过 滤 输 入 ppp 


在 EditText 右 端 设 置 输 入 提示 内 容 和 图 标 
通过 自 定义 Shape 创建 不 同 的 圆 角 按钮 … 


日 有 录 


Cl tents, 
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使 用 FloatingActionButton 创建 悬浮 按 锂 | 


以 全 屏 效 果 显 示 在 ImageView 中 的 图 像 
在 自 定 义 ImageView 中 显示 圆 形 图 像 
使 用 单 指 滑动 拖 电 ImageView 的 图 像 


使 用 SwipeRefreshLayout 切换 图 像 
使 用 AdapterViewFlipper 月 动 播放 图 像 
使 用 两 幅 图 像 定 制 ToggleButton 开关 状态 


使 用 GridView 创建 网 格 显示 多 幅 图 像 pp 
使 用 ViewPager 实现 缩放 轮 播 多 幅 图 像 pp 
使 用 Handler 实现 自动 轮 播 ViewPager ee 
使 用 ViewPager 实现 苹果 风格 的 cover flow pp 


使 用 RecyclerView 创建 水 平 瀑布 流 图 像 
以 网 格 或 列表 显示 RecyclerView 列表 项 
使 用 RecyclerView 仿 表情 包 插 入 输入 框 


使 用 CardView 显示 RecyclerView 列表 项 和 pp 


在 ListView 中 创建 图 文 结合 列表 项 
使 用 List PopupWindow 实现 下 拉 选 择 


使 用 Elevation 创建 阴影 扩散 的 控件 pp 
在 单 击 CheckBox 时 显示 波纹 扩散 效果 


使 用 目 定 义 形 状 定制 Switch 开关 状态 


匠人 
使 用 ViewSwitcher 平滑 切换 两 个 View pp 
使 用 SlidingDrawer 实现 抽 居 式 滑 动 pp 
es 
使 用 CollapsingToolbarLayout 实现 深 动 折 董 

使 用 BottomNavigationView 实现 底部 导航 ppp 
eT 
使 用 ViewOutlineProvider 创建 圆 角 控件 
,yd 
在 TextClock 中 定制 日 期 格式 pp 


使 用 RatingBar 实现 星 级 评分 


在 登录 窗口 中 使 用 SeekBar 实现 手动 校 验 PP 
文字 和 
使 用 ScaleXSpan 创建 扁平 风格 的 文字 ppp 
使 用 MaskFilterSpan 实现 文字 边 毕 模糊 ooo sroooreo oor oorree errr sereer ere er ror os 
使 用 MaskFilterSpan 实现 文字 中 心 猴 宝 pp 
使 用 MaskFilterSpan 实现 文字 整体 模糊 PP 


本 
使 用 BulletSpan 在 文本 首 字 前 添加 小 圆 大 pp 
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使 用 StrikethroughSpan 添加 文字 删除 线 ……… 
使 用 URLSpan 为 部 分 内 容 添加 超 链接 .pp 
使 用 ImageSpan 同时 显示 QQ 表情 和 文字 pp 


使 用 StyleSpan 实现 以 粗 斜体 显示 文字 … 


使 用 SuperscriptSpan 绘制 勾 股 定理 公式 pp 
使 用 SubscriptSpan 绘制 硫酸 亚 铁 分 子 式 oooeoeeoeeseoseooeoseooeoeoeoseoeeeeeeeeeeeeeeeeeee eresereeeeeereeeeee 
本 用 


使 用 ForegroundColorSpan 创建 光照 文字 


I lt Eel 全 十 而 最 搓 坝 全 


使 用 EmbossMaskFilter 创建 浮雕 文字 … 
通过 自 定义 View 在 半圆 激 上 绘制 文字 … 


ee 
en 
Ve SR 
en Vw AAA 
nn nt 
i 宇和 


A £33 
图 形 和 图 像 站 


在 日 定义 View 中 绘制 径 回 渐 变 的 图 形 … 
在 自 定 义 View 中 实现 图 像 波纹 起 伏 效果 … 


在 自 定 义 View 中 使 用 椭圆 裁 前 图像 
通过 PorterDuff 模式 增 暗 显示 两 幅 图 像 pp 
通过 PorterDuff 模式 将 图 像 裁剪 成 五 角 星 pp 
通过 PorterDuff 模式 改变 tint 属性 王 加 效果 pp 
使 用 Region 的 DIFFERENCE 实现 抠 图 功能 pp 
使 用 ShapeDrawable 裁剪 三 角形 图 像 pp 
使 用 ClipDrawable 裁剪 图 像 实现 星 级 评分 ppp 
使 用 自 定义 Drawable 实现 对 图 像 进行 圆 角 
全 用 Matrix 实现 按照 指定 方向 模 估 图 丛 sssseoeeesesoseeeassssseraeoaesresaoaeoesossanessseren 
合用 ColorMatriz 汶 图 像 渗 加 江 笋 效 困 si itd 
使 用 ColorMatrix 实现 图 像 的 加 上 暗 效 上 早 ee 
通过 自 定 义 ColorMatrix 调整 图 像 蓝 色色 调 ……………… 
使 用 RenderScript 实现 高 斯 算法 模糊 图 像 pp 
通过 像素 操作 实现 在 图 像 上 添加 光照 效果 
使 用 BitmapShader 实现 文字 线条 图 像 化 pp 
使 用 BlurMaskFilter 为 图 像 添 加 轮廓 线 pp 
使 用 PathDashPathEffect 实现 椭圆 线条 
使 用 SumPathEffect 到 加 多 种 路 径 特效 
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通过 BitmapShader 实现 以 图 像 填 充 椭 圆 pp 
使 用 ComposeShader 创建 渐变 图 像 pp 
使 用 ImageView 显示 XML 实现 的 矢量 图 形 PP 


使 用 BitmapFactory 压缩 图 像 的 大 小 


在 月 是 六 天 四 全 用 其 ae 吕 未 二 要 国民 se 
i py Cr ，weeeweeewemweweweeeweeee 
放 过 操作 根 布 局 主观 将 民 基 内 容 可 在 为 图 和 要 pene antdrdinbed cwienine 
de 
mbodlijils 让 大国 他 的 绩 网 国 eee a 
通过 采用 取 模 的 方式 实现 轮流 显示 多 幅 图 像 pp 


动画 DT ee 


使 用 ObjectAnimator 创建 上 下 振动 动画 
使 用 ObjectAnimator 实现 沿 弧 线路 径 平 移 pp 
使 用 ObjectAnimator 深 动 显示 多 幅 图 像 pp 
使 用 ObjectAnimator 实现 图 形 数字 形变 pp 
使 用 ObjectAnimator 改变 图 像 的 色相 值 pp 
使 用 AnimatorSet 组 合 多 个 ObjectAnimator ee 


使 用 TypeEvaluator 实现 颜色 过 渡 动 画 


通过 trimPathEnd 实现 动态 生成 手指 图 形 ppp 
人 SEE 
使 用 ValueAnimator 实现 分 段 转圈 动画 pp 
村 用 二 J 
使 用 ValueAnimator 实现 起 飞 转 平 飞 动画 pp 
自 定 义 TypeEvaluator 以 GIF 动画 显示 图 像 PP 
使 用 Animation 实现 图 像 围 绕 月 身 中 心 旋 转 pp 
自 定 义 Animation 实现 旋转 切换 扑克 有 牌 正 反 面 

使 用 AnimationSet 实现 组 合 多 个 不 同 的 动画 ee 
使 用 Animation 实现 按照 顺序 显示 网 格 Item ee 
使 用 windowAnimations 实现 缩放 对 话 框 窗口 pp 
使 用 AnimationDrawable 播放 多 幅 图 像 pp 
使 用 AnimationDrawable 创建 爆炸 动画 


使 用 RotateAnimation 实现 围绕 自身 中 心 旋转 


本 用 Kiphahaimaton 宣 奸 污 大 法国 动 国 0 
使 用 ScaleAnimation 创建 缩放 图 像 动画 PP 
在 ViewPager 中 实现 上 下 滑动 的 转 场 动画 ppp 
通过 下 拉手 指 实现 两 个 Activity 的 相互 切换 ppp 


在 应 用 启动 时 使 用 进 场 动画 启动 Activity 
以 左 和 人 右 出 的 动画 效果 切换 两 个 Activity 
以 收缩 扩张 的 动画 效果 切换 两 个 Activity 
使 用 转 场 动 画 Explode 切换 两 个 Activity 


使 用 转 场 动画 Slide 切换 两 个 Activity pp 


以 指定 位 置 的 转 场 动画 切换 两 个 Activity 
在 切换 Activity 时 嫩 加 缩放 动画 和 转 场 动画 
在 切换 Activity 的 转 场 动画 中 共享 多 对 元 素 


使 用 FragmentTransaction 自 定 义 转 场 动画 pp 
使 用 TransitionManager 实现 上 下 滑动 动画 和 pp 


使 用 TransitionManager 实现 用 绕 站 轴 旋 转 


使 用 TransitionManager 实现 Fade 动画 效果 pp 
使 用 TransitionManager 组 合 多 个 不 同 动画 pp 
使 用 TransitionManager 实现 单 布 局 过 渡 动 画 pp 
使 用 TransitionManager 实现 平移 过 渡 动 丽 pp 
使 用 TransitionManager 实现 缩放 部 分 图 像 … 
使 用 Transition Manager 实现 矢量 路 径 动 男 pp 
使 用 TransitionManager 同时 实现 多 种 动画 PP 
使 用 TransitionManager 实现 XML 定制 动画 pp 
使 用 TransitionManager 指定 控件 执行 动画 pp 
使 用 TransitionManager 实现 列表 项 滑 入 动画 ppp 
使 用 TransitionManager 实现 弧 线 路 径 动 田 nt 
使 用 TransitionManager 实现 裁剪 区 域 动画 ……… 人 nn 
通过 设置 和 获取 控件 的 Tag 确定 动画 过 渡 行 为 ene 
在 TransitionSet 中 指定 多 个 动画 的 执行 顺序 pp 


使 用 TransitionDrawable 透明 切换 两 幅 图 像 


使 用 AnimatedVectorDrawable 实现 转圈 动画 pp 


创建 AnimatedVectorDrawableCompat 动画 


使 用 ViewPropertyAnimator 创建 多 个 动画 pp 


自 定 义 selector 实现 以 动画 形式 改变 阴影 大 小 
使 用 ripple 标签 创建 中 心 波 纹 扩散 动画 
使 用 GLSurfaceView 实现 3D 地 球 的 自转 


使 用 MediaPlayer 播放 本 地 mp3 音乐 文件 人 
使 用 MediaPlayer 播放 本 地 mp4 视频 文件 PR 


使 用 MediaPlayer 播放 指定 网 址 的 音乐 文件 
使 用 滑 块 同步 MediaPlayer 播放 音频 的 进度 
使 用 滑 块 同步 MediaPlayer 播放 视频 的 进度 


使 用 MediaController 创建 视频 播放 控制 栏 pp 


使 用 MediaMetadataRetriever 实现 视频 截图 
使 用 MediaMetadataRetriever 获取 视频 缩 略 图 


使 用 VideoView 播放 本 地 mp4 视频 文件 eT 
使 用 VideoView 播放 指定 网 址 的 视频 文件 ppp 


使 用 MediaRecorder 录制 音频 文件 


使 用 RemoteViews 在 通知 栏 上 创建 播放 器 .pp 


在 使 用 SurfaceView 播放 视频 时 实现 模 屏 显示 
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在 选择 音乐 曲 目 窗 口中 : 选择 音乐 文件 并 播放 ee 
在 RecyclerView 中 加 载 音 乐 文 件 并 播放 和 
依次 播放 在 RecyclerView 中 的 音 首 乐 文件 eeuoeoeooeo soos eso sesso so soe ooo oodesseeee 
在 ListView 上 加 载 手机 外 存 的 音乐 文件 eee 
使 用 SoundPool 播放 较 短 的 声音 片段 


使 用 AudioManager 增 大 或 减 小 音量 
使 用 AudioManager 播放 系统 预 置 的 声音 
使 用 AudioManager 获取 和 设置 铃声 模式 


使 用 JSONObject 解析 JSON 字符 吕 pp 
使 用 JSONArray 解析 JSON 字符 训 pp 
使 用 JSONTokener 解析 JSON 字符 串 pp 
使 用 JsonReader 解析 JSON 字符 哩 pp 
使 用 JSONStringer 创建 JSON 字符 吕 pp 
使 用 JSONObject 根据 IP 显示 所 在 城市 pp 
使 用 Gson 将 数组 转换 成 JSON 字符 串 ……… 
使 用 Gson 解析 JSON 字符 吕 pp 
使 用 XmlPullParser 解析 城市 天 气 数据 … 
采用 SAX 方式 解析 XML 文件 内 容 pp 
使 用 Pattern 根据 正则 表达 式 校 验 手机 号 码 .sosoosssssssssssssssssses sosssssieoossossoseossesie 
使 用 SharedPreferences 保存 账户 和 密码 pp 
使 用 ListPreference 读 写 单 选 按钮 值 pp 
在 代码 中 获取 CheckBoxPreference 值 ne 
通过 PreferenceScreen 跳 转 到 Wifi 设置 pp 
Td 
使 用 Intent 在 Activity 之 间 传 递 图 像 和 文本 pp 
使 用 Intent 在 Activity 之 间 传 递 集 合 数据 ppp 
在 Intent 传递 数据 时 使 用 Bundle 携带 数组 
使 用 Intent 在 Service 和 Activity 之 间 传 递 数 据 … SD 
使 用 FileInputStream 和 FileOutputStream 读 取 和 保存 文本 文件 … ee. 
将 浮雕 风格 的 特效 文字 保存 为 图 像 文 件 一 ……………………… 0 

在 SD 卡 上 将 Bitmap 保存 为 PNG 图 像 文件 es 
从 手机 相册 中 选择 图 像 文件 并 裁剪 头像 ee 
在 ListView 上 加 载 手 机 外 存 的 图 像 文件 pp 
使 用 DownloadManager 下 载 网 络 文件 pp 
使 用 RandomAccessFile 实现 断 点 续 传 下 载 pp 
使 用 HttpURLConnection 下 载 图 像 文 件 PP 


使 用 QuickContactBadge 访问 联系 人 


使 用 ContentProviderOperation 增加 联系 人 … ao 


229 ”使 用 ContentProviderOperation 修改 联系 人 pp 354 
230 使 用 ContentProviderOperation 删除 联系 人 pp 356 
231 ”使 用 ContentResolver 检测 飞行 模式 的 状态 ppp 358 
232 ”使 用 ContentResolver 检测 手机 的 时 间 格 式 ppp 359 
233 ”使 用 ContentResolver 获取 所 有 短信 pp 359 
234 使 用 ContentResolver 获取 通话 记录 pp 361 
235 ”使 用 ContentResolver 获取 SD 卡 的 文件 ppp 363 
236 ”使 用 ContentResolver 改变 屏幕 亮度 值 pp 365 
237 ”使 用 ContentResolver 设置 屏幕 亮度 值 pp 366 
238 使 用 ContentResolver 检测 旋转 屏幕 功能 PP 367 
239 使 用 BroadcastReceiver 监听 来 电 电 话 号 码 68 
240 使 用 BroadcastReceiver 判断 手机 电池 是 否 正 在 充电 pp 369 
241 使 用 BroadcastReceiver 监听 屏幕 开局 或 关闭 pp 371 
242 ” 自 定 义 BroadcastReceiver 实现 短信 拦截 pp 372 
243 ”使 用 RingtoneManager 设置 手机 羡 钟 铃声 pp 373 
244 使 用 RingtoneManager 设置 手机 通知 铃声 PP 375 
245 使 用 AlarmManager 以 指定 时 间 执 行 操作 376 
246 使 用 AudioManager 获取 和 设置 音量 pp 377 
247 使 用 PowerManager 实现 屏幕 一 直 亮 着 pp 379 
248 使 用 WallpaperManager 设置 壁纸 … ee 380 
249 使 用 PackageManager 获取 支持 分 享 的 应 用 :PP 381 
250 ”使 用 WifiManager 开启 或 关闭 WiFi 信号 …… 382 
251 使 用 WifiManager 获取 IP 地 址 e384 
252 使 用 ConnectivityManager 判断 网 络 状态 PP 385 
253 使 用 BluetoothAdapter 打开 或 关闭 蓝牙 386 
254 使 用 LocationListener 获取 当前 经 纬度 值 和 pp 387 
255 ”使 用 SensorManager 获取 传感器 信息 390 
PA | 
257 ”使 用 加 速度 传感器 监听 手机 的 三 维 变化 393 
258 ”通过 传感器 实现 自动 进行 横 屏 和 竖 屏 切换 395 
259 ”使 用 setRequestedOrientation() 实现 横 屏 站 ppp 397 
260 ”根据 手机 是 模 屏 或 是 竖 屏 进行 控件 布局 .sseosoeaeeseeoeeseoeoooeooooscossooeoceooooooooooocee 398 
261 使 用 FLAG_FULLSCREEN 标志 实现 全 屏 显 示 pp 399 
262 使 用 Display 获取 屏幕 宽度 和 高 度 … e401 
263 ”使 用 StatFs 获取 内 部 总 空间 和 可 用 空间 大 小 pp 401 
264 使 用 GestureDetector 实现 纵 回 滑动 切换 pp 

265 自 定义 手机 振动 器 (Vibrator) 的 振动 模式 和 405 
266 ”使 用 SurfaceView 实现 照相 机 的 预览 功能 ppp 406 
267 ”使 用 Camera 实现 缩小 和 放大 预览 画面 .0 408 
268 ”使 用 Camera 实现 预览 时 摄像 头 手动 对 焦 ……………… 409 
270 ”使 用 Runnable 间隔 执行 重复 的 任务 .pp 413 
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271 
212 


使 用 ‘Timer 实现 促销 活动 的 倒计时 功能 gE 
使 用 Runtime 执行 系统 命令 静默 安装 应 用 包 和 
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273 
274 
219 
276 
211 
278 


使 用 腾讯 SDK 获取 授权 QQ 账户 的 简介 pp 
使 用 腾讯 SDK 实现 以 第 三 方 登录 QQ 账户 
使 用 腾讯 SDK 将 指定 文本 分 享 给 QQ 好 友 ee 
使 用 腾讯 SDK 将 本 地 图 像 发 表 到 QQ 空间 pp 
使 用 微 信 SDK 将 音乐 链接 分 享 到 朋友 圈 eee 
使 用 百度 SDK 根据 起 点 和 终点 规划 步行 线路 ee 


使 用 百度 SDK 调用 百度 地 图 App 的 驾车 导航 


使 用 百度 SDK 实现 在 地 图 中 定位 手机 位 置 PT 剖 源 曾 丰 在 站 而 而 矶 直击 丰 着 二 而 省 直击 而 覃 攻 机 丰 曾 证 在 曾 本 看 
使 用 百度 SDK 在 地 图 的 城市 之 间 绘 制 连 线 ……ee 
使 用 百度 SDK 在 地 图 上 添加 图 文 悬 浮 框 pT 避 沁 着 辣 晤 看 吉 厨 癌 后 疝 硬 司 晤 种 晤 语 
使 用 百度 SDK 在 地 图 上 添加 淡 入 动 丽 
使 用 百度 SDK 在 地 图 上 查询 指定 城市 兴趣 点 和 
使 用 百度 SDK 在 地 图 指定 范围 添加 圆 角 甜 形 ……………… 
使 用 百度 SDK 查询 指定 地 点 的 热力 图 TE 
使 用 百度 SDK 实现 隐藏 或 显示 地 名 标注 信息 ee 
使 用 百度 SDK 实现 根据 经 纬度 计算 两 地 距离 
使 用 新 浪 SDK 获取 授权 微 博 账户 的 简介 
使 用 新 浪 SDK 将 微 博 账户 简介 生成 二 维 码 ee 
使 用 新 浪 SDK 实现 搜索 指定 关键 字 的 微 博 ……eeee 
使 用 新 浪 SDK 实现 发 布 图 像 至 微 博 ooooeeeooooo ooo eso es oo sos oo oo ooooooooooosee 


UI 布局 


001 使 用 纯 Java 代码 创建 应 用 UI 界面 


此 实例 主要 通过 在 LinearLayout 线性 布局 管理 器 中 使 用 addView() 方 法 添加 控件 ,从 而 以 纯粹 
Java 代码 的 方式 实现 UI 界面 。 当 实例 运行 之 后 ,最 初 的 界面 效果 如 图 001. 1 的 左 图 所 示 。 单 击 “ 刷 
新 内 容 ” 按 钮 , 则 显示 最 新 的 日 期 信息 ,如 图 001. 1 的 右 图 所 示 。 


MySample 


elloAndroid 炫 酷 应 用 实例 集锦 ! 
当前 日 期 : Mon Jun 04 14:35:04 GMT 
08:00 2018 


刷新 内 容 


图 001.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
// setContentView(R. layout. activity main) ; 
LinearLayout myLinearLayout = new LinearLayout(this); // 创 建 线 性 布局 管理 器 
super. setContentView(myLinearLayout); // 加 载 线性 布局 管理 器 
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// 在 线性 布局 管理 器 中 垂直 布局 控件 

myLinearLayout. setOrientation(LinearLayout. VERTICAL); 

final TextView myTextView = new TextView(this); // 创 建 TextView 控件 
myTextView. setText("Hello, World! " ); // 设 置 TextView 控件 的 文本 
myTextView. setTextSize(20); // 设 置 TextView 控件 的 字体 大 小 
Button myButton = new Button(this); // 创 建 Button 控件 

myButton. setText(" 刷 新 内 容 " ); // 设 置 Button 控件 的 文本 
myButton. setTextSize(20); // 设 置 Button 控件 的 字体 大 小 


// 为 Button 控件 添加 Click 单 击 事件 响应 方法 
myButton. setOnClickListener(new View.OnClickListener() { 
@ Override 
public void onClick(View v) { 
myTextView. setText("Hello, Android 炫 酷 应 用 实例 集锦 !\n 当前 日 期 :" 
+ new java.util.Date( )); 


} 0); 
myLinearLayout. addView(myTextView); // 在 线性 布局 管理 器 中 添加 TextView 控件 
myLinearLayout. addView( myButton); // 在 线性 布局 管理 器 中 添加 Button 控件 
1 


上 面 这 段 代 码 在 MyCode\ MySample001\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,所 有 的 控件 都 使 用 new 关键 字 创 建 实例 ,然后 通过 
LinearLayout 线性 布局 管理 器 的 addView() 方 法 添加 控件 。 当 使 用 new 关键 字 创 建 控件 时 ,在 控件 
的 实例 化 方法 中 都 传人 了 一 个 this 参数 ,这 是 由 于 在 Android 中 创建 UI 控件 时 需要 传人 一 个 
Context 参数 ,Context 代表 访问 Android 应 用 环境 的 全 局 信息 的 API。 如 果 UI 控件 带 有 一 个 
Context 参数 , 则 可 以 让 UI 控件 通过 该 Context 参数 来 获取 Android 应 用 环境 的 信息 。Android 的 
大 多 数控 件 都 继承 自 View, 因 此 在 控件 实例 化 之 后 , 即 可 使 用 LinearLayout 的 addView() 方 法 向 UI 
中 添加 该 控件 。 上 默认 情况 下 ,Android 的 UI 控件 在 activity_main. xml 文件 中 以 可 视 化 的 方式 进行 设 
计 ,activity_main. xml 文件 是 一 个 完全 的 XML 格式 的 文件 ,然后 使 用 setContentView (R. layout. 
activity_main) 方 法 添加 此 XML 文件 。 如 果 界 面 比较 复杂 ,使 用 XML 文件 设计 UI 控件 应 是 最 优选 
择 ; 如 果 界 面 比较 简单 ,或 者 在 运行 时 需要 动态 调整 , 则 应 该 使 用 Java 代码 添加 控件 。 此 实例 的 完整 
项 目 在 MyCode\MySample001 文件 夹 中 。 


002 ”使 用 自 定 义 View 代 榴 布局 文件 


此 实例 主要 通过 在 自 定 义 View 中 使 用 EmbossMaskFilter 创建 浮雕 文字 ,并 直接 使 用 
setContentView() 方 法 加 载 该 自 定 义 View, 从 而 实现 代替 布局 文件 (Cactivity_main. xml) 的 效果 。 当 
实例 运行 之 后 ,将 显示 在 自 定 义 View 中 创建 的 浮雕 文字 ,如 图 002. 1 所 示 。 主 要 代码 如 下 : 


public class MainActivity extends Activity { 

Override 

protected void onCreate(Bundle savedInstanceState) { 
setContentView(new MYView(this) ); // 加 载 自 定 义 View 
super. onCreate(savedInstanceState) ; 

} 

class MyView extends View { // 创 建 自 定义 View 
private Paint myPaint; 
EmbossMaskFilter myFilter; 
public MyView(Context context) { 
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super (context); 

myPaint = new Paint(Paint.ANTI ALIAS FLAG); 
myPaint. setColor(Color. RED); 

myPaint. setStyle(Paint. Style. STROKE); 
myPaint. setStrokeWidth(64); 

myPaint. setTextSize(800); 

float[] myDirection = new float[]{1, 1, 1)}; 


float myLight = 0.05f; // 设 置 环境 光亮 度 
float mySpecular = 5; // 设 置 反射 等 级 
float myBlur = 13; // 设 置 模糊 级 别 
myFilter = new EmbossMaskFilter(myDirection, 
myLight, mySpecular, myBlur); // 自 定义 浮雕 滤 镜 
myPaint. setMaskFilter(myFilter); // 设 置 浮雕 滤 镜 的 画笔 
} 
(Override 
protected void onDraw(Canvas myCanvas) { 
super. onDraw( myCanvas ); 


Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay. getWidth( ); 
int myHeight = myDisplay. getHeight( ); 
myCanvas. drawText(" 炫 ", myWidth / 10， 
myHeight / 2 + 70, myPaint); // 显 示 浮 雕 文本 
史 兹 | 


MySample 


图 002.1 


上 和 面 这 段 代 人 码 在 MyCode\ MySample086 \app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,setContentView (new MyView (this)) 用 于 加 载 自 定义 
View, 以 显示 浮雕 文字 。 此 外 ,在 使 用 EmbossMaskFilter 时 ,需要 在 AndroidManifest. xml 文件 中 添 
加 android: hardwareAccelerated 一"false" , 即 关 闭 人 硬件 加 速 , 否 则 不 能 显示 浮雕 效果 。 此 实例 的 完 
整 项 目 在 MyCode\MySample086 文件 夹 中 ， 
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003 ”使 用 TableLayout 布局 多 个 输入 框 


此 实例 主要 通过 使 用 TableLayout 布局 管理 器 ,实现 在 其 中 以 列 对 齐 的 方式 布局 多 个 TextView 
控件 和 EditText 控件 。 当 实例 运行 之 后 ,使 用 TableLayout 布局 的 多 个 TextView 控件 和 EditText 
控件 的 效果 如 图 003. 1 所 示 。 

主要 代码 如 下 : 


< TableLayout android: id = "(@ + id/TableLayout1" 
android:layout width= "match parent” 
android:layout height = "match parent" 
android:paddingBottom = "(@dimen/activity vertical_ margin" 
android:paddingLeft = "(@dimen/activity horizontal margin" 
android:paddingRight = "(@dimen/activity horizontal margin" 
android:paddingTop = "@dimen/activity vertical margin"> 
< TableRow > Q 
<TextView android:text=" 用 户 名 :" 


hm 


- = MySample 
android: textSize = "20dp" /> 
< EditText android: layout width = "300dp" a 
android:hint = "请 输入 用 户 名 " E 晴 输 入 用 PP 名 
android: inputTYPe = " textPersonName" 密 码 : 请 输入 密码 
android: textSize = "20dp" /> 
</TableRow > 确认 密码 : 请 再 次 输入 密码 
< TableRow > 电话 号 码 : 请 输入 电话 号 码 
<TextView android:text= "密码 :" 


android:textSize = "20dp"/> 邮箱 地 址 : 请 输入 邮箱 地 址 
<EditText android:hint = "请 输入 密码 " 
android:password = "true" 
android:textSize = "20dp"/> 
</TableRow> 
< TableRow> 
<TextView android:text = "确认 密码 :" 
android:textSize = "20dp" /> 
<EditText android:hint = "请 再 次 输入 密码 " 
android: inputType = "textPassword" 
android:textSize = "20dp"/> 
</TableRow > 
< TableRow > 
<TextView android:text = "电话 号 码 :" 
android:textSize = "20dp"/> 
<EditText android:hint = "请 输入 电话 号 码 " 
android: inputType = "phone" 
android:textSize = "20dp"/> 
</TableRow > 
< TableRow > 
< TextView android:text = "邮箱 地 址 :" 
android:textSize = "20dp"/> 
< EditText android:hint = "请 输入 邮箱 地 址 " 
android: inputType = "textEmailAddress" 
android:textSize = "20dp"/> 
</TableRow > 
</TableLayout > 


003.1 


上 面 这 段 代 码 在 MyCode\MySample369\app\src\main\res\layout\activity_main. xml 文件 中 。 
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在 这 段 代 码 中 ,每 一 行 的 TextView 控件 和 EditText 控件 均 使 用 TableLayout 控件 的 TableRow 标 
签 来 控制 ,如 果 试 图 删除 TableLayout 标签 , 仅 用 TableRow 标签 , 则 应 用 在 运行 时 将 立即 前 溃 。 此 实 
例 的 完整 项 目 在 MyCode\MySample369 文件 夹 中 。 


004 使 用 TextInputLayout 管理 输入 框 提示 


此 实例 主要 通过 使 用 TextInputLayout 控件 ,实现 管理 输入 框 EditText 的 提示 信息 。 在 实例 运 
行 之 后 , 当 输 入 框 获得 焦点 时 ,提示 信息 “请 输入 用 户 名 称 ” 将 自动 滑 向 左上 和 角 , 从 而 在 原始 位 置 留 出 


功能 。 如 果 输 入 框 的 内 容 为 空白 , 则 在 单 击 “登录 ”按钮 之 后 ,在 输入 框 的 下 面 将 显示 红色 的 提示 信 
息 , 如 “用 户 密 码 不 能 是 空白 ”, 如 图 004. 1 的 右 图 所 示 。 主 要 代码 如 下 : 


Q VE TAQ%2% 下 午 3:17 | Q 四 yO dx MB 2 下 午 3:18 


请 输入 用 户 名 称 


请 输入 用 户 名 称 、 
luobin 


请 输入 用 户 密 码 


请 输入 用 户 密码 


用 户 密码 不 能 是 空白 ! 


图 004. 1 


public class MainActivity extends Activity { 

EditText myEditAccount, myEditPassword; 
TextInputLayout myAccountWrapper, myPasswordWrapper; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
myEditAccount = (EditText) findViewById(R. id. myYRccount ) ; 
myEditPassword = (EditText) findViewById(R. id. myPassword ) ; 
myAccountWrapper = (TextInputLayout) findViewById(R. id. accountWrapper) ; 
myPasswordWrapper = (TextInputLayout) findViewById(R. id. passwordWrapper); 
myAccountWrapper. setHint(" 请 输入 用 户 名 称 " ); 
myPasswordWrapper. setHint(" 请 输入 用 户 密码 " ); 

} 

public void onClickmyBtnl (View v) { // 响 应 单 击 "登录 "按钮 
String myAccount = myEditAccount. getText( ) .toString( ) ; 
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String myPassword = myEditPassword. getText( ) .toString( ) ; 
if (myAccount. length()< 1) { myAccountWrapper. setError(" 用 户 名 称 不 能 是 空白 !"); } 
else if (myPassword. length()<1) { 
myPasswordWrapper. setError(" 用 户 密码 不 能 是 空白 !"); 
} else { 
myAccountWrapper. setErrorEnabled(false); 
myPasswordWrapper. setErrorEnabled(false); 
//doLogin( ); 
FT 


上 和 面 这 段 代 码 在 MyCode\ MySample435\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myAccountWrapper. setHint(" 请 输入 用 户 名 称 ") 用 于 设 
置 输入 框 的 提示 信息 ,myAccountWrapper. setError(" 用 户 名 称 不 能 是 空 蝗 1") 用 于 设置 输入 框 的 错 
误 提 示 信 息 。 从 TextInputLayout 控件 名 称 可 以 看 出 ,该 控件 是 一 个 布局 控件 ,输入 框 EditText 控件 
则 包 右 在 其 中 ,主要 代码 如 下 : 


<LinearLayout android:layout width= "match parent" 
android:layout height = "wrap_content" 
android:layout centerInParent = "true" 
android:orientation = "vertical"> 
< android. support. design. widget. TextInputLayout 
android: id = "(@ + id/accountWrapper" 
android: layout width= "match parent" 
android:layout height = "wrap_content" 
android:layout marginTop = "4dp"> 
<EditText android:id= "(@ + id/myAccount" 
android:layout width= "match parent" 
android:layout height = "45dp" 
android:layout marginLeft = "16dp" 
android:layout marginRight = "16dp" 
android:layout marginTop = "20dp" 
android:gravity = "center_vertical" 
android:paddingLeft = "12dp"/> 
</android. support. design. widget. TextInputLayout > 
< android. support. design. widget. TextInputLayout 
android: id = "(@ + id/passwordWrapper" 
android:layout width= "match parent" 
android:layout _ height = "wrap_content" 
android:layout below = "(@ id/accountWrapper" 
android:layout marginTop = "4dp"> 
<EditText android: id= "(@ + id/myPassword" 
android:layout width= "match parent" 
android:layout height = "45dp" 
android:layout _ marginLeft = "16dp" 
android:layout marginRight = "16dp" 
android:layout marginTop = "20dp" 
android:gravity = "center_vertical" 
android: inputType = "textPassword" 
android:paddingLeft = "12dp"/> 
</android. support. design. widget. TextInputLayout > 
<Button android: id = "(@ + id/login" 
android:layout width= "match parent" 
android: layout_ height = "45dp” 
android:layout marginLeft = "20dp" 
android: layout marginRight = "20dp" 
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android:layout marginTop = "30dp" 
android:background = "(@color/colorPrimary" 
android:onClick = "onClickmyBtn1" 
android:text = "登录 " 
android:textColor = " # fff" 
android:textSize = "20dp"/> 

</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample435\app\src\main\res\layout\activity_main. xml 文件 中 。 
需要 说 明 的 是 ,在 Android Studio 中 使 用 TextInputLayout 控件 需要 在 gradle 中 引入 compile 'com. 
android. support: design: 24. 2.0' 和 compile 'com. android. support: appcompat-v7: 24. 2.0' 依 赖 项 ， 
24 是 当前 开发 环境 的 SDK 版 本 号 。 如 果 应 用 的 主题 是 android: Theme. Material. Light. 
DarkActionBar， 应 用 可 能 在 运行 时 会 骨 溃 ,因此 需要 将 app\src\main\res\values\styles. xml 文件 的 
主题 < style name 一 "AppTheme"”parent 王 "android: Theme. Material. Light. DarkActionBar"> 修 改 


为 < style name 一 " AppTheme”parent 一"Theme. AppCompat. Light. NoActionBar" > 或 < style name 一 " 
AppTheme" parent 王 "Theme。AppCompat. Light. DarkActionBar">。 此 实例 的 完整 项 目 在 MyCode\ 
MySample435 文件 夹 中 ， 


005 ”使 用 GridLayout 创建 计算 如 按键 布局 


此 实例 主要 通过 使 用 GridLayout 布局 管理 器 ,将 控件 放置 在 指定 行 和 指定 列 的 网 格 , 从 而 创建 包括 
不 同 大 小 按钮 的 计算 器 界面 。 当 实例 运行 之 后 ,包括 不 同 大 小 按钮 的 计算 器 界面 如 图 005. 1 所 示 。 
主要 代码 如 下 : 


<GridLayout android: id= "(@ + id/myGridLayout" 四 Q 9 图 口 Y 
android: layout width= "match parent" MySample 
android:layout height = "match parent" 
android: columnCount = "4" 
android: rowCount = "6"> 

<! -- 定义 一 个 横 跨 4 列 的 AutoCompleteTextView 一 一 > 1 2+56=68 
< AutoCompleteTextView android:layout width= "match parent" 
android:layout height = "180dp" 
android:layout columnSpan = "4" 
android:layout marginLeft = "2pt" 
android:layout marginRight = "2pt" 
android:background = "#fff" 
android:padding = "3pt" 
android:text ="12+56=68" 
android:textColor = "#000" 
android:textSize = "48dp"/> 
<! -- 定义 一 个 横 跨 4 列 的 Button -一 > 
< Button android:]layout width= "match parent" 
android: layout height = "70dp" 
android: layout columnSpan = "4" 
android:text = "清除 " 
android:textSize = "28dp"/> 
</GridLayout > 


005.1 


上 面 这 段 代 码 在 MyCode\MySample011\app\src\main\res\layout\activity_main. xml 文件 中 。 在 这 
段 代 码 中 ,android: columnCount 二 "4" 表 示 GridLayout 布局 管理 器 有 4 列 ,android: rowCount 王 "6" 表 示 


六 Android 炫 酷 应 用 300 例 。 实战 篇 


GridLayonut 布局 管理 器 有 6 行 , 即 此 实例 的 GridLayonut 布局 管理 吾 是 4 列 6 行 组 成 的 网 格 。android: 
layout_columnSpan 王 "4" 表 示 Button 控件 占据 GridLayout 布局 管理 器 4 列 , 即 第 二 行 全 部 列 ( 路 列 显 
示 )。 如 果 使 用 Java 代码 向 GridLayout 布局 管理 器 添加 控件 , 则 必须 在 addView() 方 法 的 第 二 个 参数 中 
指明 行 号 和 列 号 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
GridLayout myGridLayout; 
// 定 义 字 符 串 数组 ,保存 16 个 按钮 的 标题 
String[ ] nyTitle = new String[ ]{"7", "8", "9", "+", 
"aA", "5", "6", "x", 
"l"，"2"，"3"，" 
"nn, "=" n+"}; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState); 
setContentView(R. layout. activity main); 
myGridLayout = (GridLayout) findViewById(R. id.myGridLayout); 
for (int i = 0; i< myTitle. length; i++) { 
Button myButton = new Button(this); 
myButton. setText (myTitle[i]); 
myButton. setTextSize( 36); // 设 置 该 按钮 的 字号 大 小 
// 指 定 该 按钮 所 在 的 行 号 ,2 表示 从 第 3 行 开 始 计 数 
GridLayout. Spec myRow = GridLayout. spec(i / 4 + 2); 
GridLayout. Spec myColumn = GridLayout. spec(i % 4); // 指 定 该 按钮 所 在 的 列 号 
GridLayout. LayoutParams myParams = 
new GridLayout. LayoutParams(myRow, myColumn); 
myGridLayout. addView(myButton, myParams); 
| 


上 面 这 段 代 码 在 MyCode\ MySample011\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , 仅 实 现 了 将 计算 右 的 16 个 按钮 ,根据 指定 的 行 号 和 列 号 ， 
放置 在 了 GridLayout 布局 管理 央 的 第 3 行 至 第 6 行 的 16 个 单元 格 中 。 一 般 情况 下 ,使 用 
GridLayout 布局 管理 器 ,必须 设置 android: columnCount 和 android: rowCount 这 两 个 属性 ,以 定义 
网 格 大 小 。 

GridLayonut 布局 管理 器 常用 的 属性 说 明 如 下 。 

(1) android: alignmentMode 属性 ,该 属性 设置 布局 管理 右 末 用 的 对 齐 模式 。 

(2) android: columnCount 属性 ,该 属性 设置 布局 管理 需 的 列 数 。 

(3) android: columnOrderPreserved 属性 ,该 属性 设置 布局 管理 融 的 列 序号 是 否 保留 。 

(4) android: rowCount 属性 ,该 属性 设置 布局 管理 秀 的 行 数 。 

(5) android: rowOrderPreserved 属性 ,该 属性 设置 布局 管理 需 的 行 序号 是 否 保 留 。 

(6) android: useDefaultMargins 属性 ,该 属性 设置 是 否 使 用 默认 的 外 边 距 。 

GridLayonut 布局 管理 顺 中 的 子 控件 常用 的 属性 说 明 如 下 。 

(1) android: layout_column 属性 ,该 属性 设置 子 控件 在 布局 管理 器 的 第 几 列 。 

(2) android: layout_columnSpan 属性 ,该 属性 设置 子 控件 在 布局 管理 舌 中 跨 几 列 。 

(3) android: layout_gravity 属性 ,该 属性 设置 子 控件 采用 何 种 方式 占据 该 布局 管理 需 的 空间 。 

(4) android: row 属性 ,该 属性 设置 子 控件 在 布局 管理 器 的 第 几 行 。 
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(5) android: rowSpan 属性 ,该 属性 设置 子 控件 在 布局 管理 器 中 跨 几 行 。 
此 实例 的 完整 项 目 在 MyCode\MySample011 文件 夹 中 。 


006 ”使 用 RelativeLayout 按照 相 邻 关系 布局 


此 实例 主要 通过 使 用 RelativeLayout 布局 管理 项 以 相对 布局 的 方式 ,实现 将 5 张 扑克 有 牌 按照 相 
邻 关 系 摆 放 到 指定 位 置 。 当 实例 运行 之 后 ,5 张 扑 殉 牌 按照 相 邻 关系 的 摆 放 效果 如 图 006. 1 所 示 。 
主要 代码 如 下 : 


< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns :tools = "http://schemas. android. com/tools" 
android: id= "@ + id/activity_main" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:paddingBottom = "(dimen/activity vertical margin" 
android:paddingLeft = "@dimen/activity horizontal margin" 
android:paddingRight = "(dimen/activity horizontal margin" 
android:paddingTop = "@dimen/activity vertical margin" 
tools:context = "com. bin. luo. mysample. MainActivity"> 
<! -- 设置 该 控件 位 于 RelativeLayout 布局 管理 器 的 中 间 -=-> 
< TextView android: id = "(@ + id/view01" 
android:layout width = "wrap_content"” 
android:layout height = "wrap_content" 
android: layout centerInParent = "true" 
android:background = "(@mipmap/center"/> 
<! -- 设置 该 控件 位 于 view01 控件 的 上 方 --> 
< TextView android: id= "(@ + id/view02" 
android: layout width = "wrap_content" 
android: layout_ height = "wrap_content" 
android: layout above = "(@id/view01" 
android: layout alignLeft = "(0@id/view01" 
android:background = "(@mipmap/top" /> 
<! -- 设置 该 控件 位 于 view01 控件 的 下 方 -一 > 
<TextView androlid: id = "(@ + id/view03" 
androlid:layout_width = "wrap_content" 
android:layout height = "wrap_content" 
android: layout alignLeft = "(0@id/view01" 
android: layout below = "(@id/view01" 
android:background = "(@mipmap/bottom" /> 
<! -- 设置 该 控件 位 于 view01 控件 的 左边 --> 
<TextView androlid: id = "(@ + id/view04" 
android:layout width = "wrap_content" 
android:layout height = "wrap_content" 
android: layout alignTop = "(@id/view01" 
android: layout toLeftOf = "(0@id/view01" 
android:background = "(@mipmap/left"/> 
<! -- 设置 该 控件 位 于 view01 控件 的 右边 -一 > 
< TextView androlid: id = "(@ + id/view05" 
android:layout width = "wrap_content” 
android:layout height = "wrap_content" 
android: layout alignTop = "(@id/view01" 
android: layout toRightOf = "(@id/view01" 
android:background = "(@mipmap/right"/> 
</RelativeLayout > 


MySample 


上 面 这 段 代 码 在 MyCode\MySample010\app\src\main\res\layout\activity_main. xml 文件 中 。 
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在 这 段 代 码 中 , @ mipmap/right、 @ mipmapy/left、Q mipmapybottom 、Q mipmap/top、\ @ mipmap/ 
center 是 将 5 张 jpg 格式 的 扑克 牌 图 像 文件 拖 放 到 res\mipmap 节点 (目录 ) 生 成 的 图 像 资源 ,用 于 设 
置 5 个 TextView 控件 的 背景 图 像 。5 个 TextView 控件 的 相 令 关系 则 直接 由 layout_alignTop、 
layout_toRightOf 等 属性 指定 。 下 面 是 RelativeLayout 布局 管理 器 的 子 控 件 与 相 邻 位 置 有 关 的 属性 
说 明 。 

(1) android: layout_centerHorizontal 属性 ,该 属性 表示 是 否 水 平 居 中 。 

(2) android: layout_centerVertical 属性 ,该 属性 表示 是 否 垂 直 居 中 。 

(3) android: layout_centerInParent 属性 ,该 属性 表示 是 否 相 对 于 父 控件 完全 居中 。 

(4) android: layout_alignParentBottom 属性 ,该 属性 表示 是 否 贴 紧 父 控 件 的 下 边缘 。 

(5) android: layout_alignParentLeft 属性 ,该 属性 表示 是 否 贴 紧 父 控 件 的 左边 绿 。 

(6) android: layout_alignParentRight 属性 ,该 属性 表示 是 否 贴 紧 父 控 件 的 右边 绿 。 

(7) android: layout_alignParentTop 属性 ,该 属性 表示 是 否 贴 紧 父 控件 的 上 边缘 。 

(8) android: layout_alignWithParentI{fMissing 属性 ,该 属性 表示 如 果 找 不 到 对 应 的 兄弟 控件 , 则 
以 父 控件 作 参 照 物 。 

(9) android: lavyout below 属性 ,该 属性 表示 在 某 控 件 的 下 方 。 

(10) android: layout_above 属性 ,该 属性 表示 在 某 控件 的 上 方 。 

(11) android: layout_toLeftOf 属性 ,该 属性 表示 在 某 控 件 的 左边 。 

(12) android: layout_toRightOf 属性 ,该 属性 表示 在 某 控件 的 右边 。 

(13) android: layout_alignTop 属性 ,该 属性 表示 本 控件 的 上 边缘 和 某 控件 的 上 边缘 对 齐 。 

(14) android: layout_alignLeft 属性 ,该 属性 表示 本 控件 的 左边 缘 和 某 控件 的 左边 缘 对 齐 。 

(15) android: layout_alignBottom 属性 ,该 属性 表示 本 控件 的 下 边缘 和 某 控件 的 下 边缘 对 齐 。 

(16) android: layout_alignRight 属性 ,该 属性 表示 本 控件 的 右边 绿 和 某 控件 的 右边 缘 对 齐 。 

(17) android: layout_marginBottom 属性 ,该 属性 表示 离 某 控件 底 边 缘 的 距离 。 

(18) android: layout_marginLeft 属性 ,该 属性 表示 离 革 控件 左边 缘 的 距离 。 

(19) android: layout_marginRight 属性 ,该 属性 表示 离 某 控件 右边 缘 的 距离 。 

(20) android: layout_marginTop 属性 ,该 属性 表示 离 某 控 件 上 边缘 的 距离 。 

此 实例 的 完整 项 目 在 MyCode\MySample010 文件 夹 中 ， 


007 ”使 用 ConstraintLayout 在 右 下 角 布 局 


此 实例 主要 通过 使 用 android. support. constraint. ConstraintLayout 布局 控件 ,实现 将 布局 里 面 
的 一 个 子 控件 放置 在 男 一 个 子 控 件 的 右 下 角 。 当 实例 运行 之 后 ,克林顿 头像 (ImageView 控件 ) 放 置 
在 奥巴马 头像 (ImageView 控件 ) 的 右 下 角 , 如 图 007.1 所 示 。 主 要 代码 如 下 : 


<?xml] version = "1.0" encoding = "utf 一 8"?> 

< android. support. constraint. ConstraintLayout 
xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns:app = "http://schemas. android. com/apk/res — auto" 
android:id= "(@ + id/activity_main" 
android: layout width= "match parent" 
android:layout height = "match parent" 
android:paddingLeft = "20dp" 
android:paddingTop = "100dp"> 

< ImageView android:id= "(@ + id/myImagel" 
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android: layout width= "wrap content" 
android:layout height = "wrap_content" 
android: src = "(@mipmap/myimagel"/> 
<! -- 实现 居中 效果 -一 > 
<! —— app:layout_ constraintBottom toBottomOf = "parent" ——> 
<! —— app:layout constraintLeft toLeftOf = "parent” 一 一 > 
<! —— app:layout constraintRight toRightOf = "parent” 一 一 > 
<! —— app:layout constraintTop toTopOf = "parent” 一 一 > 
< ImageView android:layout width= "wrap_content” 
android: layout height = "wrap_content” 
android: src = "(@mipmap/myimage2" 
app: layout constraintLeft toRightOf ="(@ + id/myImagel" 
app: layout constraintTop toBottomOf = "(@ + id/myImagel" /> 
</android. support. constraint. ConstraintLayout > 


上 面 这 段 代 码 在 MyCode\MySample363\app\src\main\res\ 
layout\activity_main. xml 文件 中 。 在 这 段 代 码 中 ,app: layout_ 
constraintLeft_ toRightOf 二"@ 十 id/myImagel" 表 示 当 前 子 控 件 
myImage2 在 子 控件 myImagel 的 右边 。app: layout_constraintTop 
_ toBottomOf=="@ 十 id/mylImagel" 表 示 当 前 子 控件 myImage2 在 
子 控件 myImagel 的 下 边 。ConstraintLayout 支持 的 具有 相对 定位 
的 属性 组 合 说 明 如 下 。 

(1) layout_constraintLeft toLeftOf 属性 ,该 属性 表示 当前 控件 
的 左边 在 某 控 件 的 左边 , 即 左 对 齐 。 

(2) layout_constraintLeft_toRightOf 属性 ,该 属性 表示 当前 控 
件 的 左边 在 某 控 件 的 右边 。 

(3) layout_constraintRight_toLeftOf 属性 ,该 属性 表示 当前 控 
件 的 右边 在 某 控件 的 左边 。 

(4) layout_constraintRight_toRightOf 属性 ,该 属性 表示 当前 
控件 的 右边 在 某 控件 的 右边 , 即 右 对 齐 。 

(5) layout_constraintTop toTopOf 属性 ,该 属性 表示 当前 控件 图 007.1 
的 上 边 和 某 控件 的 上 边 对 齐 , 即 上 对 齐 。 

(6) layout_constraintTop_toBottom0Of 属性 ,该 属性 表示 当前 控件 的 上 边 在 某 控件 的 下 边 。 

(7) layout_constraintBottom_toTop0Of 属性 ,该 属性 表示 当前 控件 的 下 边 在 某 控 件 的 上 边 。 

(8) layout_constraintBottom_toBottom0Of 属性 ,该 属性 表示 当前 控件 的 下 边 在 某 控 件 的 下 边 , 即 
下 对 齐 。 

需要 说 明 的 是 ,在 Android Studio 中 使 用 ConstraintLayout 之 前 ,需要 在 app 下 的 gradle 文件 中 
添加 ConstraintLayout 依赖 项 ,如 : 


dependencies { 
compile 'com.android. support. constraint:constraint -~ layout:1.0.2'} 


此 实例 的 完整 项 目 在 MyCode\MySample363 文件 夹 中 ， 


008 ”使 用 TableLayout 拉 伸 控件 填充 容 媳 


此 实例 主要 通过 设置 TableLayout 布局 管理 器 的 stretchColumns 属性 ,实现 拉 伸 指定 列 中 的 控 
件 以 填充 TableLayout 布局 管理 器 的 剩余 空间 。 当 实例 运行 之 后 ,由 于 指定 了 TableLayout 布局 管 
理 需 的 stretchColumns 属性 是 第 二 列 ,因此 *B 图 ( 拉 伸 )” 占 用 的 空间 最 大 ,如 图 008. 1 所 示 。 

主要 代码 如 下 : 


<! -一 定义 表格 布局 ,第 二 列 允 许 拉 伸 -一 > 
< TableLayout android:layout width= "match parent" 
android:layout height = "match parent" 
android: stretchColumns = "1"> 
< TableRow > 
< TextView android:layout width= "wrap_content” 
android: layout_ height = "500dp" 
android:background = " # eee0a0" 
android:gravity = "center" 
android:text = "A 图 (默认 )" 
android:textAlignment = "center"/> 
<TextView android:layout width= "wrap_content" i B 图 ( 拉 促 ) 
android:layout height = "fill parent" 
android:background = " # 00aa00" 
android:gravity = "center" 
android:text = "B 图 ( 拉 伸 )"/> 
< TextView android:layout width= "wrap_content” 
android:layout height = "fil1_parent” 
android:background = " # O00FFFF" 
android:gravity = "center" 
android:text = "C 图 (默认 )"/> 
</TableRow > 008.1 
</TableLayout > 


MySample 


上 面 这 段 代 码 在 MyCode\MySample007\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: stretchColumns 王 "1" 表 示 拉 伸 第 二 列 的 控件 (如 果 有 剩余 空间 ) ,android 
stretchColumns 王 "0" 则 表示 拉 伸 第 一 列 的 控件 (如 果 有 剩余 空间 ) 。 如 果 有 多 列 控件 需要 拉 伸 , 则 应 
在 属性 值 中 使 用 逗号 隔 开 列 索 引 , 如 android: stretchColumns 三 "1,2" 表 示 同 时 拉 伸 第 二 列 和 第 三 列 
的 控件 (如 果 有 剩余 空间 )。stretchColumns 是 布局 管理 器 TableLayout 的 属性 ,用 于 设置 允许 被 拉 
伸 的 列 序号 ,TableLayout 继承 自 LinearLayout,; 因 此 它 完全 可 以 支持 LinearLayout 的 属性 。 此 实例 
的 完整 项 目 在 MyCode\MySample007 文件 夹 中 。 


009 ”使 用 TableLayout 缩小 控件 适应 容器 


此 实例 主要 通过 设置 TableLayout 布局 管理 器 的 collapseColumns 属性 ,实现 在 屏幕 不 能 显示 全 
部 控件 时 , 折 和 县 (隐藏 ) 指 定 的 控件 。 当 实例 运行 之 后 ,效果 如 图 009. 1 的 左 图 所 示 ; 该 图 表示 由 于 屏 
莫 限 制 , 不 能 完全 显示 A、B、C、D、E、F、G 图 (控件 ), 因 此 折 芋 (隐藏 )B 图 和 C 图 ,但 是 这 样 做 之 后 就 
产生 了 一 点 剩余 空间 ,因此 又 拉 伸 了 A 图 以 填 满 所 有 空间 。 默 认 情 况 下 ,如 果 不 指定 折 全 控件, 当 屏 
幕 不 能 完全 显示 所 有 控件 时 , 则 将 按照 顺序 自动 折 有 登 (隐藏 ) 最 后 的 控件 ,如 图 009. 1 的 右 图 所 示 。 
主要 代码 如 下 : 


MySample MySample 


| [| 


图 009.1 


“| 定义 TableLayout 布局 , 折 琶 1、2 列 , 拉 伸 第 0 列 开 


< TableLayout 


< TableRow > 


< TextView 


< TextView 


< TextView 


< TextView 


< TextView 


android:layout width= "match parent" 
android: layout _ height = "match parent" 
android:collapseColumns = "1,2" 
android: stretchColumns = "0"> 


android: 
android: 


android 


android 


android 


layout width= "wrap_content" 
layout height = "500dp" 


:background = " # eee0a0" 
android: 
android: 


gravity = "center" 


text = "A 图 (默认 )"/> 


:layout width= "wrap_content" 
android: 
android: 
android: 
android: 
android: 
android: 


layout height = "fill parent" 
background = "# 00aa00" 
gravity = "center" 

text = "B 图 (默认 )"/> 

layout width= "wrap_content" 
layout height = "fill parent" 


:background = " # O00AAFF" 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 


gravity = "center" 

text = "C 图 (默认 )"/> 
layout_width = "wrap_content” 
1ayout_height = "fill parent" 
background = " #5 CCFFFF" 
gravity = "center" 

text = "D 图 (默认 )"/> 

layout width= "wrap_content" 
layout height = "fill parent" 
background = " #5 BBOOFF" 
gravity = "center" 


text = "E 图 (默认 )"/> 
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< TextView android:layout width= "wrap_content” 
android:layout height = "fill parent" 
android:background = " #FFFF00" 
android:gravity = "center" 
android:text = "F 图 (默认 )"/> 

<TextView android:layout width= "wrap_content" 
android:layout height = "fill] parent" 
android:background = " 井 00EFF00" 
android:gravity = "center" 
android:text = "G 图 (默认 )"/> 

</TableRow ></TableLayout > 


上 和 面 这 段 代 码 在 MyCode\MySample008\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 人 码 中 ,android: collapseColumns 三 "1,2" 表 示 折 芋 (隐藏) 第 二 列 和 第 三 列 中 的 子 控 件 。 
collapseColumns 是 布局 管理 器 TableLayout 的 属性 ,用 于 指定 需要 折 芭 的 列 序号 ,TableLayout 继承 
自 LinearLayout。TableLayout 另外 两 个 比较 有 特色 的 属性 是 stretchColumns 和 shrinkColumns， 
shrinkColumns 属性 用 于 设置 允许 被 收缩 的 列 序号 ,stretchColumns 属性 用 于 设置 允许 被 拉 伸 的 列 序 
号 。 此 实例 的 完整 项 目 在 MyCode\MySample008 文件 夹 中 。 


010 ”使 用 LinearLayout 纵 回 居中 对 齐 控件 


此 实例 主要 通过 设置 LinearLayout 线性 布局 管理 峰 的 gravity 属性 ,实现 多 个 控件 在 屏幕 中 的 不 
同位 置 对 齐 显 示 。 当 实例 运行 之 后 , 单 击 “ 靠 左 对 齐 ” 按 钮 , 则 3 幅 图 像 靠 左 对 齐 的 效果 如 图 010. 1 的 
左 图 所 示 。 单 击 “ 居 中 对 齐 ” 按 钮 , 则 3 幅 图 像 居 中 对 齐 的 效果 如 图 010. 1 的 右 图 所 示 。 单 击 “ 靠 右 对 
齐 ” 按 钮 , 则 3 幅 图 像 将 靠 右 对 齐 显示 。 


图 010. 1 
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<LinearLayout xmlns:android= "http://schemas.android. com/apk/res/android" 
android: id = "(@ + id/myLinearLayout" 
android:layout width= "match parent" 


android:layout height = "match parent" 
android:gravity = "left |center vertical" 
android:orientation = "vertical"> 


< ImageView android: 
android: 
android: 
< ImageView android: 
android: 
android: 
< ImageView android: 
android: 
android: 


</LinearLayout > 


layout width = "96dp" 
layout height = "96dp" 

src = "(mipmap/box"/> 
layout width = "96dp" 
layout height = "96dp" 

src = "(mipmap/pen"/> 
layout width = "96dp" 
layout height = "96dp" 
src= "(mipmap/airship"/> 


UI 布局 全 ; 


上 面 这 段 代 码 在 MyCode\MySample004\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: gravity 一 "centerj| center_vertical "表示 在 水 平方 句 和 垂直 方 回 均 居 中 显示 在 
LinearLayout 中 的 控件 , 即 3 幅 图 像 居 中 对 齐 。android: gravity 一 "left| center_vertical" 表 示 在 水 平 
方 问 上 靠 左 对 齐 .在 垂直 方 回 均 居 中 显示 在 LinearLayout 中 的 控件 , 即 3 幅 图 像 靠 左 对 齐 。android: 
orientation 王 "vertical" 表 示 3 幅 图 像 在 一 列 中 显示 ,如 果 设 置 android: orientation 王 "horizontal", 则 
3 幅 图 像 将 在 一 行 中 显示 。 默 认 情 况 下 ,设置 LinearLayout 线性 布局 管理 髓 的 控件 对 齐 方式 均 通 过 
在 XML 文件 中 操作 gravity 属性 实现 ; 如 果 需 要 使 用 Java 代码 动态 修改 控件 的 对 齐 方式 , 则 应 该 使 用 
setGravity() 方 法 ,主要 代码 如 下 : 


public class MainActivity extends Activity { 
LinearLayout myLinearLayout; 


(QOverride 


protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 


setContentView(R. layout. activity main); 


myLinearLayout = (LinearLayout) findViewById(R. id.myLinearLayout); 


} 


public void myBtnLeftClick(View v) { // 响 应 单 击 " 靠 左 对 齐 " 按钮 
myLinearLayout. setGravity(Gravity. LEFT | Gravity. CENTER_VERTICRL ) ; 

} 

public void myBtnCenterClick(View v) { // 响 应 单 击 " 居 中 对 齐 " 按 钮 
myLinearLayout. setGravity(Gravity. CENTER | Gravity.CENTER VERTICAL); 

} 

public void myBtnRightClick(View v) { // 响 应 单 击 " 靠 右 对 齐 " 按 钮 


myLinearLayout. setGravity(Gravity. RIGHT | Gravity. CENTER_VERTICRL ) ; 


} } 


上 面 这 段 代 码 在 MyCode\MySample004\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代码 中 ,setGravity() 方 法 中 的 参数 即 是 gravity 属性 值 ,该 属性 值 可 以 有 多 个 ,使 用 
“|” 符 号 分 隔 ; 支持 的 属性 值 包括 : TOP、AXIS X _ SHIFT、AXIS Y_SHIFT 、LEFT 、RIGHT NO _ 
GRAVITY .RELATIVE HORIZONTAL GRAVITY MASK 、_ BOTTOM、HORIZONTAL GRAVITY _ 
MASK VERTICAL GRAVITY MASK.AXIS PULL AFTER 、CLIP VERTICAL CENTER CENTER 


HORIZONTAL \CENTER_VERTICAL.\CLIP_HORIZONTAL \FILL_HORIZONTAL .DISPLAY _CLIP 
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HORIZONTAL DISPLAY _CLIP_ VERTICAL., RELATIVE LAYOUT _ DIRECTION.,\ START \FILL_ 
VERTICAL。 需 要 注意 的 是 : 当 采 用 LinearLayout 线性 布局 管理 器 管理 控件 时 ,如 果 其 中 的 多 个 控件 无 
法 在 单行 中 同时 显示 , 则 LinearLayout 不 会 换行 显示 后 面 未 显示 的 控件 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample004 文件 夹 中 。 


011 使 用 LinearLayout 按 权 重 分 配 控件 空间 


此 实例 主要 通过 为 LinearLayout 线性 布局 管理 器 的 子 控件 设置 不 同 的 layout_weight 属性 ( 权 
重 ) 值 ,从 而 实现 按 比例 调节 子 控件 的 空间 占 比 。 当 实例 运行 之 后 ,“B 图 ”的 layout_weight 属性 ( 权 
重 ) 值 最 大 ,因此 占用 的 空间 最 大 ,如 图 011. 1 所 示 。 

主要 代码 如 下 : 


< LinearbLayout } 
android: layout width =" fill parent" MySample 
android:layout height = "fill parent" 
android:orientation = "horizontal"> 
< TextView android:layout width = "wrap content" 
android: layout height = "fill] parent" 
android: layout weight = "1" 
android:background = "# ff0000" 
android:gravity = "center horizontal" 
android:text = "A 图"/> 
< TextView android:layout width= "wrap content" 
android:layout height = "fill parent" 
android: layout weight ="4" 
android:background = "#00aa00" 
android:gravity = "center horizontal" 
android:text = "B 图 "/> 
< TextView androlid:1layout_width = "wrap_content" 
android:layout height = "fil1_parent” 011.1 
android: layout weight ="1" 
android:background = " # O00FFFF" 
android:gravity = "center horizontal" 
android:text = "C 图 "/> 
< TextView android:layout width= "wrap content" 
android:layout height = "fill parent" 
android: layout weight = "1" 
android:background = "#aaaa00" 
android:gravity = "center_ horizontal" 
android:text = "D 图 "/> 
</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample005\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: layout_weight 二 "4" 表 示 该 控件 的 权重 是 4, 它 是 一 个 相对 数 ,总 是 相对 相关 
的 其 他 控件 而 言 ,此 处 权重 为 4 的 控件 实际 表示 对 额外 空间 的 分 配 权 重 是 : 4/ (4 十 1 十 1 十 1)= 二 57%。 
在 Android 中 ,layout_weight 属性 表示 子 控件 对 LinearLayout 额外 空间 的 划分 ,需要 注意 的 是 额外 
二 字 , 先 有 额外 的 空间 ,然后 才 可 以 按 比 例 将 其 分 配给 设置 了 layout_weight 的 子 View, 所 以 如 果 
LinearLayout 的 layout_width 或 layout_height 属性 设置 了 WRAP _ CONTENT . 则 LinearLayout 就 
没有 额外 的 空间 ,因此 子 控件 的 layout_weight 就 没有 用 处 。 此 实例 的 完整 项 目 在 MyCodeA\ 
MySample005 文件 夹 中 。 
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012 ”使 用 ConstraintLayout 平分 剩余 空间 


此 实例 主要 通过 使 用 android. support. constraint. ConstraintLayout 布局 控件 ,实现 使 三 个 控件 
ImageView 平分 在 水 平方 向 上 的 剩余 空间 。 当 实例 运行 之 后 ,奥巴马 头像 .克林顿 头像 .普京 头像 将 
平分 水 平方 向 的 剩余 空间 ,如 图 012. 1 所 示 。 


图 012. 1 


主要 代码 如 下 : 


<?xml version = "1.0" encoding = "utf 一 8"?> 
< android. support. constraint.ConstraintLayout 

xmlns:android = "http://schemas. android. com/apk/res/android" 

xmlns:app = "http://schemas.android. com/apk/res — auto" 

android: id = "(@ + id/activity main" 

android:layout marginTop = "200dp" 

android: layout width= "match parent" 

android:layout height = "wrap_content"> 

< ImageView android:id= "(@ + id/myImagel" 
android: layout_width = "wrap_content" 
android: layout height = "wrap_content" 
android: src = "(@mipmap/myimagel" 
app: layout constraintLeft toLeftOf = "parent" 
app: layout constraintRight toLeftOf = "(0 + id/myImage2" /> 
< ImageView android:id= "(@ + id/myImage2" 

android: layout_width = "wrap_content" 
android: layout height = "wrap_content 
android: src = "(@mipmap/myimage2" 
app: layout_constraintLeft_ toRightOf = "(@ + id/myImagel” 
app: layout constraintRight toLeftOf ="(@ + id/myImage3" /> 
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< ImageView android:id= "(@ + id/myImage3" 
android:layout width= "wrap_content" 
android:layout height = "wrap content" 
android: src = "(@Qmipmap/myimage3" 
app: layout_constraintLeft toRightOf = "(@ + id/myImage2" 
app: layout_ constraintRight toRightOf = "parent" /> 
</android. support. constraint. ConstraintLayout > 


上 和 面 这 段 代码 在 MyCode\MySample364\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代码 中 ,app: layout_ constraintLeft_ toRightOf 二"@ 十 id/myImage2" 表示 当前 子 控件 
myImage3 在 子 控件 myImage2 的 右边 。app: layout_constraintRight。 toRightOf 王 "parent" 表 示 当 
亲子 控件 myImage3 与 ConstraintLayout 布局 控件 右 对 齐 。 默 认 情 况 下 ,由 于 3 个 子 控件 ImageView 
在 水 平方 回 形 成 一 个 链 式 布局 ,因此 它们 会 平分 水 平方 回 的 剩余 空间 。 需 要 说 明 的 是 ,在 Android Studio 
中 使 用 ConstraintLayonut 之 前 ,需要 在 app 下 的 gradle 文件 中 添加 ConstraintLayout 依赖 项 ,如 下 : 


dependencies { 
compile 'com. android. support. constraint:constraint -~ layout:1.0.2'} 


此 实例 的 完整 项 目 在 MyCode\MySample364 文件 夹 中 ， 


013 ”使 用 ConstraintLayout 无 间 际 布局 控件 


此 实例 主要 通过 使 用 android. support. constraint. ConstraintLayout 布局 控件 ,并 设置 子 控件 的 
app: layout_constraintHorizontal_chainStyle 属性 为 packed, 实 现 清除 子 控件 之 间 的 间 际 。 当 实例 运 
行 之 后 ,奥巴马 头像 .克林顿 头像 .普京 头像 3 个子 控 件 之 间 没 有 间 际 ,水 平 剩余 空间 全 部 分 配 到 奥 巴 
马 头 像 和 普京 头像 与 ConstraintLayout 布局 控件 (屏幕 左右 两 侧 ) 的 毗邻 处 ,如 图 013. 1 所 示 。 


= 中 所 呈 x 已 


MySample 


图 013.1 
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<?xml] version= "1.0" encoding = "utf 一 8"?> 
< android. support. constraint. ConstraintLayout 
xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns:app = "http://schemas. android. com/apk/res — auto" 
android:id= "(@ + id/activity_main" 
android: layout width = "match parent" 
android:layout height = "wrap_content" 
android:layout marginTop = "200dp"> 
< ImageView app: layout constraintHorizontal chainStyle = "packed" 
android: id= "(@ + id/myImagel" 
android: layout width= "wrap content" 
android: layout height = "wrap_content" 
android: src = "(Wnmipmap/myimagel" 
app: layout constraintLeft toLeftOf = "parent" 
app: layout constraintRight toLeftOf = "(© + id/myImage2"/> 
< ImageView android: id= "(@ + id/myImage2" 
android:layout width= "wrap_content" 
android: layout_height = "wrap_content" 
android: src = "(@mipmap/myimage2" 
app: layout constraintLeft toRightOf ="(@ + id/myImagel" 
app: layout constraintRight toLeftOf ="(@ + id/myImage3" /> 
< ImageView android:id= "(@ + id/myImage3" 
android:layout width= "wrap_content" 
android:layout height = "wrap_content" 
android: src = "(@mipmap/myimage3" 
app: layout constraintLeft toRightOf ="(@ + id/myImage2" 
app: layout constraintRight toRightOf = "parent"/> 
</android. support. constraint. ConstraintLayout > 


上 面 这 段 代码 在 MyCode\MySample365\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,app: layout_constraintHorizontal chainStyle 王 "packed" 表 示 view 之 间 紧 挨 着 显示 ， 
并 且 全 体 居 中 显示 ; 如 果 删 除 此 行 代 码 , 在 图 013. 1 中 ,奥巴马 头像 与 克林顿 头像 之 间 .克林顿 头像 与 
普京 头像 之 间 就 会 出 现 颖 际 。 需 要 说 明 的 是 ,在 使 用 ConstraintLayout 之 前 ,需要 在 app 下 的 gradle 
文件 中 添加 ConstraintLayout 依赖 项 ,如 下 : 


dependencies { 
compile 'com.android. support. constraint:constraint -~ layout:1.0.2') 


此 实例 的 完整 项 目 在 MyCode\MySample365 文件 夹 中 ， 


014 使 用 TabLayout 和 适 配 费 创建 选项 证 


此 实例 主要 通过 使 用 TabLayout 和 PagerAdapter, 从 而 创建 滑动 选项 卡 。 当 实例 运行 之 后 , 单 击 
“京东 ”选项 卡 , 则 在 选项 卡 中 加 载 的 京东 (手机 网 站 ) 页 面 效 果 如 图 014. 1 的 左 图 所 示 ; 单 击 “360? 选 
项 卡 , 则 在 选项 卡 中 加 载 的 360( 手 机 网 站 ) 页 面 效 果 如 图 014. 1 的 右 图 所 示 。 单 击 其 他 选项 卡 将 实现 
类 似 的 功能 。 


CCo > Android 可 可 a0og 。 实战 篇 


习 6D 手机 卫士 
弘 扰 拦截 | 超级 加 速 | 支付 安全 


立即 下 载 


国力 O 风 SB 


全 部 产品 360 搜 索 360 商 城 360 游 戏 


A 一 
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手机 卫士 ” 手机 助手 


主要 代码 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<LinearLayout xmlns:android = "http://schemas.android. com/apk/res/android" 
android: id = "(@ + id/activity main" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:orientation = "vertical"> 
< android. support. design. widget. TabLayout 
android: id= "@ + id/myTabLayout" 
android:layout width= "match parent" 
android:layout height = "wrap_content"/> 
< android. support. v4. view. ViewPager 
android: id= "(@ + id/myViewPager" 
android:layout width= "match parent” 
android:layout_ height = "wrap_content" /> 
</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample677\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代码 中 ,TabLayonut 控件 主要 用 于 管理 选项 卡 的 标签 ,ViewPager 控件 则 用 于 管理 选项 卡 的 视 
图 (网 站 页 面 )。 两 者 可 以 通过 TabLayout 的 setupWithViewPager (ViewPager) 关 联 起 来 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
private TabLayout myTabLayout; 
private ViewPager myViewPager; 
private LayoutIinflater myInflater; 
private List < String> myTitles = new ArrayList < String >(); // 选 项 卡 标 题 集合 
private List <View> myViews = new ArrayList <View>(); // 选 项 卡 页 面 集 合 
Override 
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protected void onCreate( Bundle savedInstanceState) { 
super. onCreatel( savedInstanceState); 
setContentView(R. layout.activity main); 
myViewPager = (ViewPager) findViewById(R. id.myViewPager); 
myTabLayout = (TabLayout) findViewById(R. id.myTabLayout); 
InitView( ); 
// 添 加 选项 卡 标题 
myTitles.add(" 京 东 "); 
myTitles.add(" 天 猫 "); 
myTitles.add("360" ); 
myTitles.add( "搜狐 "); 
myTitles.add(" 百 度 "); 
MyAdapter myAdapter = new MyAdapter(myViews,myTitles); 
myViewPager. setAdapter (myAdapter); // 给 ViewPager 设置 适配器 
// 将 TabLayout 和 ViewPager 关联 起 来 
myTabLayout. setupWithViewPager(myViewPager ) ; 


private class MyWebViewClient extends WebViewClient { 

(Override 

public boolean shouldOverrideUrlLoading(WebView view, String url) { 
View. loadUrl (ur]l); 
return true; 


上 
private void InitView() { 
myInflater = LayoutInflater.from(this); 
String[ ] myWebsites = {"https://www. jd.com","http://www.tmall.com", 
"http: //www. 360.cn","http: //www. sohu. com" ," http: //www. baidu. com" }; 
for(int i=1;i<=5;i+t+){ 
View myView = myInflater. inflate(R. layout. myitem, null); 
WebView myWebView = (WebView) myView.findViewById(R. id.myWebView); 
myWebView. loadUrl (myWebsites[i- 1]); 
myWebView. getSettings( ) . setJavaScriptEnabled(true); 
myWebView. setWebViewClient(new MyWebViewClient( )); 
myViews. add( myView); 
} }} 


上 面 这 段 代 码 在 MyCode\ MySample677\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myViewPager. setAdapter(myAdapter) 表 示 使 用 自 定义 
适 配 严 MyAdapter 管理 选项 卡 的 标题 (标签 ) 和 内 容 ( 即 web 页 面 )。 目 定义 适 配 天 MyAdapter 的 主 
要 代码 如 下 : 


public class MyAdapter extends PagerAdapter { 

private List < View> myViews; 

private List < String> myTitles; 

public MyAdapter(List <View> mViewList, List < String> titles) { 
this.myViews = mViewList; 
myTitles = titles; 

} 

(Override 

public int getCount( ) { return myViews. size(); } 

(Override 
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public boolean isViewFromObject(View view, Object object){ return view == object;} 
Override 
public Object instantiateItem(ViewGroup container, int position) { 
container. addView(myViews. get (position) ); // 添 加 选项 卡 
return myViews. get (position); 
} 
(QOverride 
public void destroyItem(ViewGroup container, int position, Object object) { 
container. removeView(myViews. get(position) ); // 删 除 选 项 卡 
} 
(Override 
public CharSequence getPageTitlel( int position) {return myTitles. get(position); } 
} 


上 和 面 这 上段 代码 在 MyCode\ MySample677\app\src\ main\java\com\bin\luo\mysample\ 
MyAdapter. java 文件 中 。 在 MainActivity. java 文件 中 ,myView 二 myInflater. inflate (R. layout. 
myitem,null) 用 于 在 选项 卡 视 图 上 加 载 myitem 布局 ,myitem 布局 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< LinearLayout 
xmlns:android = "http://schemas. android. com/apk/res/android" 
android: id= "(@O + id/activity main" 
android:layout width= "match parent" 
android:layout height = "match parent"> 
< WebView android: id= "@ + id/myWebView" 
android:layout width= "match parent" 
android:layout height = "match parent"/> 
</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample677\app\src\main\res\layout\myitem. xml 文件 中 。 需 要 
说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 23. 3. 0， 
依赖 项 。 此 外 ,访问 网 络 需要 相关 的 权限 ,因此 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission android: name 一 "android. permission. INTERNET"/ > 权限 。 如 果 在 编译 和 安装 时 出 现 
错误 ,请 注意 检查 主题 是 否 有 错误 , 即 可 能 需要 将 app\src\main\res\values\styles. xml 文件 的 主题 
< style name="AppTheme" parent 二 "android: Theme. Material. Light. DarkActionBar" > 修改 为 
< style name="AppTheme" parent 二 "Theme. AppCompat. Light. DarkActionBar">。 此 实例 的 完整 


项 目 在 MyCode\MySample677 文件 夹 中 。 


015 ”使 用 TabLayout 和 Fragment 创建 选项 卡 


此 实例 主要 通过 使 用 TabLayout 和 Fragment, 实 现 创建 选项 卡 风格 的 界面 。 当 实例 运行 之 后 ， 
单 击 “ 亲 密 的 陌生 人 ”标签 , 则 将 显示 该 标签 对 应 的 选项 卡 , 效 果 如 图 015. 1 的 左 图 所 示 。 单 击 “ 善 意 
的 谎言 ?标签 , 则 将 显示 该 标签 对 应 的 选项 卡 ,效果 如 图 015. 1 的 右 图 所 示 。 单 击 “ 光 末 岁 月 ”标签 ,也 
将 显示 该 标签 对 应 的 选项 卡 。 
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图 015. 1 
主要 代码 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<LinearLayout xmlns:android= "http://schemas.android. com/apk/res/android" 
xmlns:app = "http://schemas.android. com/apk/res — auto" 
android: layout width= "match parent" 
android:layout height = "match parent" 
android:orientation = "vertical"> 
< android. support. design. widget. TabLayout 
android: id= "(@ + id/myTabLayout" 
android:layout width= "match parent" 
android:layout height = "55dp" 
app:tabGravity = "center" 
app: tabIndicatorColor = "(@color/colorAccent" 
app:tabMode = "scrollable" 
app :tabSelectedTextColor = "(@color/colorPrimaryDark" 
app :tabTextColor = "(Qcolor/colorPrimary"/> 
< android. support. v4. view. ViewPager 
android: id= "(@ + id/myViewPager" 
android:layout width= "match parent" 
android:layout height = "match parent"/> 
</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample357\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,TabLayout 控件 主要 用 于 管理 选项 卡 的 标签 ,ViewPager 控件 则 用 于 管理 选项 卡 的 视 
图 (电影 海报 图 像 )。 当 在 Activity 中 加 载 TabLayout 时 , Activity 通常 需要 从 AppCompatActivity 
继承 ,主要 代码 如 下 : 


public class MainActivity extends AppCompatActivity { 
String[ ] myTitle = {" 亲 密 的 陌生 人 "”,， "善意 的 谎言 "， "光荣 岁月 "} ; 
String[ ] myData = {" 亲 密 的 阴 生 人 ", "善意 的 谎言 "， "光荣 岁月 "}; 


少 Android 炫 酷 应 用 300 例 。 实战 篇 


TabLayout myTabLayout; 

ViewPager myViewPager; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _ main) ，; 
initView( ) ; 

} 

private void initView() { 
myTabLayout (TabLayout) findViewById(R. id. myTabLayout); 
myViewPager = (ViewPager) findViewById(R. id.myViewPager); 
myViewPager. setAdapter (new FragmentPagerAdapter(getSupportFragmentManager( ) ){ 


(Override 
public CharSequence getPageTitlel( int position) { 
return myTitle[position % myTitle. length]; // 此 方法 负责 管理 选项 卡 标签 
} 
(@ Override 
public Fragment getItem( int position) { // 创 建 Fragment 并 返回 


TabFragment myTabFragment = new TabFragment( ); 
myTabFragment. setTitle(myData[ position % myTitle. length]); 
return myTabFragment; 
} 
(Override 
public int getCount() {return myTitle. length; } 
bs 
myTabLayout. setupWithViewPager (myViewPager); // 将 ViewPager 关联 TabLayout 
myTabLayout. setOnTabSelectedListener(new TabLayout. OnTabSelectedListener() { 
(Override 
public void onTabSelected(TabLayout. Tab tab) { 
myViewPager. setCurrentItem(tab. getPosition( ) ); // 切 换 ViewPager 
} 
(@ Override 
public void onTabUnselected(TabLayout. Tab tab) { } 
(@ Override 
public void onTabReselected(TabLayout. Tab tab) { } 


He 


上 面 这 段 代 码 在 MyCode\ MySample357\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,TabFragment myTabFragment 二 new TabFragment() 用 
于 在 ViewPager 中 显示 控件 (此 处 是 电影 海报 图 像 )。TabFragment 类 的 主要 代码 如 下 : 


public class TabFragment extends Fragment { 
private String myTitle; 
String[ ] myMovies = {" 亲 密 的 陌生 人 ", "善意 的 谎言 ", "光荣 岁月 "}; 
public void setTitle(String title) { this.myTitle = title; } 
(Override 
public View onCreateView(LayoutInflater inflater, 
ViewGroup container, Bundle savedInstanceState) { 
ImageView myImageView = new ImageView(getContext()); 
myImageView. setScaleTypel( ImageView. ScaleType. FIT XY); 
if(myTitle. contains(myMovies[0]))t{ 
myImageView. setImageResource(R. mipmap. myimage!l ); 
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}else if(myTitle. contains(myMovies[1])){ 


myImageView. setImageResource(R. mipmap.myimage2 ) ; 
}else if(myTitle. contains(myMovies[2])){ 
myImageView. setImageResource(R.mipmap.myimage3 ); 
} 
return mylImageView; 


1 


上 面 这 上段 代码 在 MyCode\ MySample357\app\src\ main\java\com\bin\luo\mysample\ 
TabFragment. java 文件 中 。 需 要 说 明 的 是 ,TabLayout 是 design 库 提 供 的 控件 , 它 既 可 以 单独 使 用 ， 
也 可 以 配合 ViewPager 控件 来 使 用 ,在 Android Studio 中 使 用 此 控件 时 只 需要 在 gradle 中 引入 
compile 'com. android. support: design: 23. 3.0' 即 可 使 用 。 具 体操 作 步 骤 如 下 : 在 Android Studio 
环境 中 打开 app\build. gradle 文件 ,在 dependencies 中 添加 compile 'com. android. support: design: 
23. 3.0', 此 时 ,在 其 上 面 将 立即 弹出 一 个 黄色 的 提示 框 , 单 击 右 侧 的 “Sync Now” 超 链接 ,Android 
Studio 将 会 自动 添加 此 依赖 项 。 

另外 ,由 于 默认 创建 MainActivity 继承 自 Activity, 如 果 像 实例 这 样 MainActivity 继承 自 
AppCompatActivity, 则 需要 将 app\src\main\res\values\styles. xml 文件 的 主题 < style name = 
"AppTheme" parent 王 "android: Theme. Material. Light. DarkActionBar"> 修 改 为 < style name = 
"AppTheme" parent 二 "Theme. AppCompat. Light. DarkActionBar"> ,否则 可 能 无 法 创建 此 应 用 。 

此 实例 的 完整 项 目 在 MyCode\MySample357 文件 夹 中 。 


016 ”使 用 FrameLayout 创建 纵 癌 选项 卡 


此 实例 主要 通过 动态 创建 FrameLayout, 从 而 创建 纵 回 风格 的 选项 卡 。 当 实例 运行 之 后 , 单 击 
“百度 ”选项 卡 , 则 效果 如 图 016. 1 的 左 图 所 示 ; 单 击 “京东 ?选项 卡 , 则 效果 如 图 016. 1 的 右 图 所 示 。 
单 击 其 他 选项 卡 将 实现 类 似 的 功能 。 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
String[ ] myTitles = { "百度 "， "京东 ", "天 猫 ", "搜狗 ", "微软 ", "新 浪 ", "淘宝 "}; 
(Override 
protected void onCreate(Bundle savedInstanceState){ 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity_ main) ; 
GetFragmentManager( ) .beginTransaction( ) .add(R. id.myFrame, 
new WebFragment( ) ) . commit( ); 
LinearLayout myLayout = (LinearLayout )findViewById(R. id. myLayout); 
for(int i = 0; i< myTitles. length; i++ ){ 
FrameLayout myFrameLayout = new FrameLayout (this); 
myFrameLayout. setLayoutParams (new FrameLayout. LayoutParams( 
ViewGroup. LayoutParams. MATCH PARENT, ViewGroup.LayoutParams. WRAP CONTENT)); 
// 自 动 生成 myFrameLayout 帧 布局 ID 值 
myFrameLayout. setId(View. generateViewId( ) ); 
// 动 态 添加 myFrameLayout 布局 (选项 卡 的 标签 ) 
myLayout. addView( myFrameLayout ); 
TitleFragment myTitleFragment = new TitleFragment( ); 
Bundle myBundle = new Bundle( ); 
myBundle. putString("myTitle", myTitles[i]); 
myTitleFragment. setArguments(myBundle); // 传 递 选 项 卡 的 标签 
getFragmentManager( ) .beginTransaction( ). 
add(myFrameLayout. getId(),myTitleFragment).commit( ); 
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上 和 面 这 段 代 码 在 MyCode\ MySample813\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 , myTitleFragment 二 new TitleFragment() 中 的 
TitleFragment 是 自 定 义 类 ,用 于 设置 选项 卡 的 标题 。TitleFragment 类 的 主要 内 容 如 下 : 


public class TitleFragment extends Fragment { 
String[ ] myTitles = {" 百 度 ", "京东 ", "天 猫 ", "搜狗 ", "微软 ", "新浪 ", "淘宝"}; 
String[ ] myUrls = {"http://www.baidu.com", "http://www. jd.com", 
"http://www. tmall. com", "http://www. sougou. com", "http://www.microsoft. com", 
"http://www. sina. com. cn", "http://www. taobao. com'" } ; 

String myTitle; 

TextView myTextView; 

List< String> myTitleList = new ArrayList(), myUrlList = new ArrayList( ); 
(Override 
public void setArguments(Bundle args) { myTitle = args. getString( "myTitle"); } 
(Override 
public View onCreateView(LayoutInflater inflater， 

ViewGroup container, Bundle savedInstanceState) { 
myTextView = new TextView(inflater.getContext()); 
myTextView. setLayoutParams(new LinearLayout. LayoutParams( ViewGroup. 
LayoutParams. MATCH PARENT, ViewGroup.LayoutParams. WRAP CONTENT)); 

myTextView. setText(myTitle); 

myTextView. setTextSize( 18); 

myTextView. setPadding(0, 25, 0, 25); 

myTextView. setGravity(Gravity. CENTER HORIZONTAL); 

return myTextView; 


第 1 章 Uh 局 《2?》 


(QOverride 
public void onActivityCreated(Bundle savedInstanceState) { 
super. onActivityCreated( savedInstanceState) ; 
for (int i = 0; i< myTitles. length; i++) { // 循 环 初始 化 数据 
myTitleList.add(myTitles[ i]); 
myUrlList.add(myUrls[ i]); 
} 
myTextView. setOnClickListener(new View. OnClickListener() { // 响 应 单 击 标签 
(Override 
public void onClick(View v) { 
// 通 过 FragmentManager 对 象 获取 指定 Fragment 
WebFragment myWebFragment = 
(WebFragment) getFragmentManager( ).findFragmentById(R. id.myFranme); 
Bundle myBundle = new Bundle( ); 
// 将 与 标题 对 应 的 网 址 通过 Bundle 传 入 WebFragment 
myBundle. putString(" myUrl1", 
myUrlList. get(myTitleList. indexOf (myTextView. getText( )))); 
myWebFragment. setArguments(myBundle); 
myWebFragment. reloadView( ); // 重 新 加 载 页 面 
1 


上 面 这 段 代 码 在 MyCode\ MySample813\app\src\ main\java\com\bin\luo\mysample\ 
TitleFragment. java 文件 中 。 在 这 段 代 码 中 ,myWebFragment 一 (WebFragment) getFragmentManager(). 
findFragmentBylId(R. id. myFrame) 中 的 WebFragment 是 自 定 义 类 ,用 于 加 载 WebView 控件 。 
WebFragment 类 的 主要 内 容 如 下 : 


public class WebFragment extends Fragment { 
String myUrl = "http://www.baidu. com" ; 
WebView myWebView; 
Override 
public void setArguments(Bundle args) { myUrl = args.getString("myUrl" ); } 
(Override 
public View onCreateView(LayoutInflater inflater, 
ViewGroup container, Bundle savedInstanceState) { 
myWebView = new WebView(inflater.getContext()); 
myWebView. getSettings(). setJavaScriptEnabled(true) ; 
myWebView. loadUrl(myUr] ); 
myWebView. setWebViewClient(new WebViewClient( )); 
return myWebView; 
} 
public void reloadView() { myWebView. loadUrl(myUrl); } 
} 


上 面 这 段 代 码 在 MyCode\ MySample813\app\src\main\java\com\bin\luo\ mysample\ 
WebFragment. java 文件 中 。 需 要 说 明 的 是 ,访问 网 络 需 要 相关 的 权限 ,因此 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 三 " android. permission. 


INTERNET"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample813 文件 夹 中 。 
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017 ”使 用 TabHost 创建 横向 选项 卡 


此 实例 主要 通过 使 用 TabHost 控件 ,实现 创建 选项 卡 风格 的 界面 。 当 实例 运行 之 后 ,选择 “黎明 
之 战 ”选项 卡 的 效果 如 图 017. 1 的 左 图 所 示 ,选择 “权力 之 眼 ” 选 项 卡 的 效果 如 图 017. 1 的 右 图 所 示 。 
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图 017. 1 


<LinearLayout androlid:1layout_width = "match parent" 
android:layout height = "match parent"> 
< TabHost android:layout width= "match parent" 
android:layout height = "match parent"> 
<! -- 定义 第 一 个 选项 卡 的 内 容 --> 
<LinearLayout android:id= "(@ + id/tab01" 
android: layout width= "fill parent" 
android:layout height = "fill] parent" 
android:background = "#121212"> 
< ImageView android:layout width= "wrap_content" 
android:layout height = "wrap_content" 
android:layout gravity = "center" 
android:layout margin = "50dp" 
android:src = "(mipmap/myimagel"/> 
</LinearLayout > 
<! -- 定义 第 二 个 选项 卡 的 内 容 --> 
<LinearLayout android:id= "(@ + id/tab02" 
android:layout width= "fill parent" 
android:layout height = "fill] parent 
android:background = " 井 121212"> 
< ImageView android:layout width= "wrap content" 


android:layout height = "wrap_content" 
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android:layout gravity = "center" 
android:layout margin = "50dp" 
android:src = "(@mipmap/myimage2"/> 
</LinearLayout > 
<! -- 定义 第 三 个 选项 卡 的 内 容 --> 
<LinearLayout android:id= "(@ + id/tab03" 
android:layout width= "fill] _ parent” 
android:layout height = "fill parent" 
android:background = "#121212"> 
< ImageView android:layout width= "wrap_content" 
android:layout height = "wrap_content" 
android:layout_ gravity = "center" 
android:layout margin = "50dp" 
android: src = "(mipmap/myimage3"/> 
</LinearLayout ></TabHost ></LinearLayout > 


上 面 这 段 代码 在 MyCode\MySample283\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 , TabHost 控件 主要 实现 了 设置 每 个 选项 卡 的 内 容 , 每 个 选项 卡通 第 还 需要 使 用 
TabHost 的 addTab() 方 法 进行 添加 才能 进行 有 效 的 管理 ,主要 代码 如 下 : 


public class MainActivity extends TabActivity{ 
(Override 
public void onCreate(Bundle savedInstanceState){ 
super. onCreate( savedInstanceState); 
TabHost myTabhost = getTabHost(); 
// 加 载 TabHost 布局 
LayoutInflater. from(this). inflate(R. layout.activity main, 
myTabhost. getTabContentView(), true); 
// 添 加 第 一 个 标签 页 
myTabhost. addTab( myTabhost. newTabSpec(" tabl" ) 
. setIndicator(" 黎 明之 战 " ). setContent(R. id.tab01) ); 
// 添 加 第 二 个 标签 页 
myTabhost. addTab(myTabhost. newTabSpec(" tab2" ) 
.SetIndicator(" 权力 之 眼 " ). setContent(R. id.tab02)); 
// 添 加 第 三 个 标签 页 
myTabhost. addTab( myTabhost. newTabSpec(" tab3" ) 
. setIndicator(" 圆梦 巨人 "). setContent(R. id. tab03)); 
} 


上 面 这 段 代 码 在 MyCode\ MySample283\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,public class MainActivity extends TabActivity 表示 当前 
承载 TabHost 的 Activity 继承 自 TabActivity ;而 不 是 常用 的 Activity, 否 则 就 无 法 使 用 getTabHost () 方 
法 。myTabhost. addTab(myTabhost. newTabSpec ("tab1l1"). setIndicator(" 黎 明之 战 "). setContent 
(R. id. tab01)) 表 示 向 TabHost 添加 R. id. tab01 选项 卡 并 设置 其 标签 为 “黎明 之 战 ?。 此 实例 的 完整 
项 目 在 MyCode\MySample283 文件 夹 中 。 


018 ”使 用 AbsoluteLayonut 实现 平移 控件 


此 实例 主要 通过 设置 在 AbsoluteLayout 布局 管理 器 中 的 子 控件 坐标 ,并 使 用 定时 器 不 断 改变 坐 
标 值 ,从 而 使 子 控件 产生 不 断 移动 的 动画 效果 。 当 实例 运行 之 后 ,降落 伞 (TextView 控件 ) 位 于 屏幕 
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的 左上 角 ,如 图 018. 1 的 左 图 所 示 ; 立即 沿 着 对 角 线 回 屏 幕 右 下 角 移 动 ,如 图 018. 1 的 右 图 所 示 ,并 且 
周而复始 , 永 不 停歇 。 


MySample MySample 


(hb) 


图 018.1 
主要 代码 如 下 : 


< AbsoluteLayout android:layout width= "match parent" 
android:layout height = "match parent"> 
<! -一 使 用 绝对 定位 定义 TextView 控件 -一 > 
< TextView android:id = "(@ + id/myTextView" 
android:layout width = "wrap_content” 
android:layout height = "wrap_content” 
android: layout_ x= "0dp" 
android:layout y= "200dp" 
android:background = "(mipmap/myimagel"/> 
</AbsoluteLayout > 


上 和 面 这 段 代 码 在 MyCode\MySample012\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: layout_x 二 "0dp" 和 android: layout_y 一 "200dp" 用 于 设置 TextView 控件 
(降落 伞 ) 的 左上 和 角 坐 标 , 它 表示 控件 位 于 AbsoluteLayout 布局 管理 需 的 左上 角 。dp 是 长 度 ( 距 离 ) 单 
位 ,Android 主要 文 持 如 下 常用 的 单位 。 

(1) px( 像 素 ) ,每 个 px 对 应 屏幕 上 的 一 个 像素 。 

(2) dip 或 dp(device independent pixels, 设 备 独立 像素 ) ,一 种 基于 屏幕 密度 的 抽象 单位 。 在 每 
in160 点 的 屏幕 上 ,1dip= 王 1px; 但 是 随 着 屏幕 密度 的 改变 ,dip 与 px 的 换算 会 发 生变 化 。 

(3) sp(scaled pixels ,比例 像 素 ) ,主要 处 理 字 体 的 大 小 ,可 以 根据 用 户 的 字体 大 小 首选 项 进行 
缩放 。 

(4) in( 英 寸 ) ,标准 长 度 单位 。 

(5) mm( 毫 米 ) ,标准 长 度 单位 。 
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(6) pt( 磅 ) ,标准 长 度 单 位 ,1/72in。 
如 果 需 要 使 用 Java 代码 动态 改变 控件 的 坐标 值 , 则 应 该 使 用 setLayoutParams() 方 法 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
public TextView myTextView; 
public static int x = 0; 
public static int y = 0; 
Handler myHandler = new Handler() { 
public void handleMessage(Message msg) { 
if (msg. what == 0xl23) { 
AbsoluteLayout. LayoutParams params = 
new AbsoluteLayout. LayoutParams(256，256，x++ 当 756, yt+% 756); 
myTextView. setLayoutParams(params); 


} 
super. handleMessage(msg) ; 


Ti 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myTextView = (TextView) findViewById(R. id.myTextView); 
new Timer(). schedule(new TimerTask() { // 周 期 性 地 改变 控件 的 坐标 
@Override 
public void run() { myHandler. sendEmptyMessage(0x123); } 
}, 0, 10); 
}} 


上 面 这 上段 代码 在 MyCode\ MySample012\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,定时 项 每 隔 10ms 调用 handleMessage() 方 法 改变 
TextView 控件 (背景 图 像 是 降落 伞 ) 的 坐标 值 。AbsoluteLayout. LayoutParams(256,256, x 十 十 
756，y 十 十 % 756) 中 的 两 个 256 分 别 表示 降落 们 图像 的 宽度 和 高 度 ,x 十 十 % 756 和 yy 十 十 % 756 
表示 TextView 控件 (降落 伞 ) 的 坐标 值 。myTextView. setLayoutParams(myParams) 即 是 根据 改变 
后 的 xy 值 设置 TextView 控件 (降落 伞 ) 的 坐标 值 。 大 多 数 情 况 下 ,使 用 AbsoluteLayonut 布局 管理 
做 可 能 不 是 最 优选 择 ,因为 运行 Android 应 用 的 手机 屏幕 大 小 、 分 辩 率 千差万别 ,使 用 绝对 布局 不 可 
能 兼顾 所 有 不 同 的 手机 ,因此 AbsoluteLayonut 布局 管理 器 已 经 不 在 推荐 使 用 之 列 , 但 是 在 需要 绝对 
定位 的 情况 下 ,AbsoluteLayout 也 是 一 个 不 错 的 选择 。 此 实例 的 完整 项 目 在 MyCode\MySample012 
文件 夹 中 。 


019 ”使 用 FrameLayout 实现 闪烁 控件 


此 实例 主要 通过 使 用 FrameLayout 布局 管理 器 二 加 其 中 的 子 控件 ,并 使 用 定时 器 不 断 改 变 其 颜 
色 , 从 而 实现 闪烁 控件 的 效果 。 当 实例 运行 之 后 ,不 同 大 小 的 实心 矩形 的 背景 颜色 将 不 断 改 变 ,形成 
闪烁 ,效果 如 图 019. 1 所 示 。 
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MySample 


主要 代码 如 下 : 


MySample 


图 019. 1 


<! -一 使 用 FrameLayout 布局 管理 器 实现 到 加 6 个 TextView 控件 -一 > 


<FrameLayout android:layout width= "wrap_content” 


android:layout height = "wrap_content"> 


<! -- 依次 定义 6 个 TextView 控件 , 先 定义 控件 位 于 底层 ,后 定义 控件 位 于 上 层 

< TextView android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 


< TextView 


android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 


< TextView 


android: 
android: 


< TextView 


android: 
android: 
android: 


android: 


id= "(@ + id/myView01" 

layout width= "wrap_content" 
layout_height = "wrap_content" 
layout gravity = "center" 
width = "300dp" 

height = "300dp" 

background = "#f00"/> 

id= "(@ + id/myView02" 

layout width= "wrap_content" 
layout height = "wrap_content" 
layout gravity = "center" 
width= "260dp" 

height = "260dp" 

background = "#0f0"/> 

id= "(@ + id/myView03" 
layout_width = "wrap_content" 
layout height = "wrap_content” 
layout gravity = "center" 
width = "220dp" 

height = "220dp" 

background = "#00f"/> 

id= "(@ + id/myView04" 
layout _ width = "wrap_content" 
layout height = "wrap_content" 
layout gravity = "center" 


android:width = "180dp" 
android:height = "180dp" 
android:background = "#ff0"/> 
< TextView android:id= "@ + id/myView05" 
android: layout_width = "wrap_content" 
android: layout height = "wrap_content" 
android:layout gravity = "center" 
android:width = "140dp" 
android: height = "140dp" 
android:background = "#f0f"/> 
<TextView android: id= "(@ + id/myView06" 
android: layout_ width = "wrap_content" 
android:layout height = "wrap_content" 
android: layout gravity = "center" 
android:width = "100dp" 
android: height = "100dp" 
android:background = " 井 0ff"/> 
</FrameLayout > 


yc 


上 面 这 段 代码 在 MyCode\MySample009\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,6 个 TextView 控件 按照 顺序 ,其 宽度 和 高 度 依次 变 小 ,形成 宝塔 形 的 琶 加 ; 
FrameLayout 布局 管理 器 在 这 里 只 是 容器 , 它 为 其 中 的 子 控件 创建 了 一 个 空白 的 区 域 ( 称 为 一 帧 ) ,每 
个 子 控件 占据 一 帧 ,这些 帧 根据 gravity 属性 执行 自动 对 齐 。 实 例 在 实现 党 虹 灯 的 动态 闪烁 效果 时 ， 
主要 是 通过 使 用 定时 央 按 照 顺 序 不 断 改变 子 控件 的 背景 颜色 , 由 于 这 些 子 控件 的 大 小 不 同 , 因 此 各 级 


子 控件 的 背景 颜色 交 蔡 改变 ,形成 内 烁 效果 。 
主要 代码 如 下 : 
public class MainActivity extends Activity { 


private int myCurrentColor = 0; 
final int[] myColors = new int[]{ // 定 义 一 个 颜色 数组 


Color. rqb{(122 122. 0}) ,Color. rob(0,. 122, 122}),Color. rgb(122, 0. 0), 
Color. rod2n5 255 OQ) Color robtD 255 295 CoOlor. ron(255 O00 O01: 

// 此 处 的 6 个 ID 是 在 activity_main. xml 文件 中 添加 的 TextView 的 ID 

final int[ ] myIDs = new int[ ]{R. id.myView01, R. id.myView02, R. id. myView03, 

R. id.myView04, R. id.myView05, R. id.myView06}; 

TextView[ ] myViews = new TextView[ myIDs. length ] ; 

Handler myHandler = new Handler() { 

public void handleMessage(Message msg) { 

if (msg.what == 0xl23) { 
for (int i = 0; i< myIDs. length; i++) { 


myViews[i]. setBackgroundColor(myColors[ (i+ myCurrentColor) % myIDs. length] ); 


} 
myCurrentColor++; 
} 
super. handleMessage(msg); 
} }; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
for (int i = 0; i< myIDs. length; i++) { 
myViews[i] = (TextView) findViewById(myIDs[i]); 
} 
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new Timer( ) . schedule(new TimerTask() { // 周 期 性 改变 控件 的 背景 颜色 
(Override 
public void run() { 
// 发 送 消息 通知 系统 改变 6 个 TextView 控件 的 背景 色 
myHandler. sendEmptyMessage( 0x123); 
} }, 0, 300); 
者 ， 


上 面 这 段 代 码 在 MyCode\ MySample009\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample009 文件 夹 中 。 


020 ”上 自 定义 FrameLayout 创建 翻 页 卷 边 动 团 


此 实例 主要 通过 在 自 定义 FrameLayout 中 使 用 quadTo() 方 法 绘制 贝 塞 尔 曲线 ,从 而 实现 图 书 翻 
页 的 卷 边 动画 效果 。 当 实例 运行 之 后 ,在 右上 (或 下 ) 半 部 分 使 用 手指 从 右 回 左 滑动 , 则 书页 从 右上 
(或 下 ) 角 产生 卷 边 效果 ,如 图 020. 1 的 左 图 所 示 ; 松 开 手指 ,日 动 显示 下 一 页 ,如 图 020. 1 的 右 图 所 
示 。 在 左上 (或 下 ) 半 部 分 使 用 手指 从 左 向 右 滑 动 , 则 书页 从 左上 (或 下 ) 角 产生 卷 边 效果 ; 松 开 手指 ， 
自动 显示 上 一 页 。 
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图 020. 1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
public void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity_main); 
PageView myPageView = (PageView) findViewById(R. id.myPageView); 
myPageView. setAdapter (new PageAdapter(this)); 
Ll 
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上 面 这 上段 代码 在 MyCode\ MySample784\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPageView 二 (PageView) findViewById (R. id. 
myPageView) 中 的 PageView 是 以 FrameLayout 为 基 类 创建 的 自 定 义 类 ,关于 该 自 定 义 类 的 主要 代 
人 码 请 参考 源 代码 中 的 MyCode\ MySample784\app\src\main\java\com\bin\luo\ mysample\ 
PageView. java 文件 。myPageView. setAdapter(new PageAdapter(this)) 表 示 使 用 PageAdapter 为 
PageView 创建 数据 适配器 ,PageAdapter 类 的 主要 代码 如 下 : 


public class PageAdapter extends BaseAdapter { 
Context myContext; 
Integer[ ] myImages = {R.mipmap.myimagel2, R.mipmap.myimage34, 
R. mipmap. myimage56}; // 总 共 3 幅 图 像 ,代表 6 页 
public PageAdapter(Context context) { myContext = context; } 
(Override 
public int getCount() { return myImages. length; } 
(Override 
public Object getItem( int position) { return myImages[ position]; } 
(Override 
public long getItemId( int position) { return position; } 
Override 
public View getView( int position, View convertView, ViewGroup parent) { 
ViewGroup myLayout; 
if (convertView == null) 
myLayout = (ViewGroup) LayoutInflater. 
from(myContext). inflate(R. layout. myitem, null); 
else myLayout = (ViewGroup)convertView; 
ImageView myImageView = (ImageView) myLayout. findViewById(R. id.myImageView); 
myImageView. setImageResource(myImages[ position|); 
return myLayout; 
图 


上 面 这 段 代 码 在 MyCode\ MySample784\app\src\main\java\com\bin\luo\ mysample\ 
PageAdapter. java 文件 中 。 在 这 段 代 码 中 , myLayout 二 (ViewGroup) LayoutInflater。 from 
(myContext). inflate(R. layout. myitem，null) 用 于 根据 myitem 布局 创建 PageAdapter 的 每 个 Item 
容器 ,这 里 只 有 一 幅 图 像 ( 书 页 )。myitem 布局 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<LinearLayout xmlns:android= "http://schemas.android. com/apk/res/android" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:baselineAligned = "false" 
android:orientation = "horizontal"> 
< ImageView android:id= "(0@ + id/myImageView" 
android:layout width= "match parent" 
android: layout height = "wrap_content" 
android: scaleType = "fitXY" /> 
</LinearLayout > 


上 上面 这 段 代 码 在 MyCode\MySample784\app\src\main\res\layout\myitem. xml 文件 中 。 此 实 
例 的 完整 项 目 在 MyCode\MySample784 文件 夹 中 。 
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021 在 TextView 中 创建 室 心 文字 


此 实例 主要 通过 设置 TextView 控件 的 shadowColor shadowRadius shadowDx、shadowDy 等 属 
性 ,从 而 创建 指定 颜色 的 空心 文字 。 当 实例 运行 之 后 ,创建 的 粉色 空心 文字 效果 如 图 021. 1 所 示 。 


MySample 


图 021. 1 
主要 代码 如 下 : 


<! 一 -创建 空心 文字 --> 

< TextView android: layout width = "wrap content" 
android: layout height = "wrap content" 
android: shadowColor = " (QO color/colorAccent" 
android: shadowDx = " 0" 
android: shadowDy = " 0" 


android: shadowRadius = "15” 
android:text =" 炫 酷 应 用 实例 " 
android: textColor = " # fff" 
android: textSize ="52sp" /> 


上 和 面 这 段 代 码 在 MyCode\MySample330\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代码 中 ,android: shadowRadius 用 于 设置 阴影 的 模糊 程度 ,该 值 越 大 ,阴影 越 模糊 。android : 
shadowDx 用 于 设置 阴影 在 水 平方 向 的 偏 移 。android: shadowDy 用 于 设置 阴影 在 垂直 方 同 的 偏 移 。 
android: shadowColor 用 于 设置 阴影 的 颜色 。 此 实例 的 完整 项 目 在 MyCode\MySample330 文件 
夹 中 。 


022 ”在 TextView 中 实现 上 文 下 图 的 布局 


此 实例 主要 实现 了 在 drawable 中 添加 图 像 资 源 , 并 设置 TextView 控件 的 android: 
drawableBottom 属性 为 该 图 像 资源 ,实现 在 TextView 控件 的 下 端 显 示 图 像 、 上 端 显示 文字 的 效果 。 
当 实 例 运 行 之 后 ,在 TextView 下 端 显示 图 像 、 上 端 显示 文字 的 效果 如 图 022. 1 所 示 。 


MySample 


重庆 市 人 民 大 礼堂 位 于 渝中 区 人 民 路 
学 田 湾 ， 于 1951 年 6 月 破土 兴建 ， 
1954 年 4 月 竣工 ， 是 一 座 仿 古 民 族 建 
筑 群 ， 也 是 重庆 独 具 特 色 的 标志 建筑 
物 之 一 。 建 筑 气势 雄伟 ， 人 金碧辉煌 ， 
是 中 国 传统 宫殿 建筑 风格 与 西方 建筑 
的 大 跨度 结构 巧妙 结合 的 杰作 ， 以 其 
非凡 的 建筑 艺术 旨 声 中 外 。 


图 022.1 
主要 代码 如 下 : 


< TextView android:layout width= "match parent" 
android:layout height = "wrap_content" 
android: drawableBottom = " (Q drawable/myimage" 
android:text = "重庆 市 人 民 大 礼堂 位 于 渝中 区 人 民 路 学 田 湾 ,于 1951 年 6 月 破土 兴建 ,1954 年 4 
月 竣工 ,是 一 座 仿古 民族 建筑 群 ,也 是 重庆 独 具 特 色 的 标志 建筑 物 之 一 .建筑 气势 雄伟 ,金碧辉煌 ,是 中 国 传统 宫 
殿 建 筑 风格 与 西方 建筑 的 大 跨度 结构 巧妙 结合 的 杰作 ,以 其 非凡 的 建筑 艺术 蔓 声 中 外 ." 
android:textSize = "20dp"/> 


上 面 这 段 代 码 在 MyCode\MySample021\app\src\main\res\layout\activity_main. xml 文件 中 。 
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在 这 段 代 码 中 ,android: drawableBottom="(@ drawable/myimage" 表 示 将 图 像 myimage 放置 在 文本 
的 Bottom( 下 端 ) ,myimage 是 图 像 资源 ID, 它 是 在 回 drawable 添加 图 像 资源 时 根据 图 像 文件 名 自动 
生成 的 。 如 果 需 要 将 图 像 放 置 在 TextView 控件 的 上 上端 、 左边 .右边 , 则 应 该 设置 该 控件 的 
drawableTop ,drawableLeft .drawableRight 等 属性 。drawableLeft 属性 表示 在 TextView 控件 的 文 
本 左边 显示 指定 图 像 ，drawableRight 属性 表示 在 TextView 控件 的 文本 右边 显示 指定 图 像 ; 
drawableTop 属性 表示 在 TextView 控件 的 文本 上 问 显 示 指 定 图 像 。drawableEnd 属性 表示 在 
TextView 控件 的 文本 结尾 显示 指定 图 像 。drawableStart 属性 表示 在 TextView 控件 的 文本 开始 显 
示 指 定 图 像 。TextView 控件 的 drawablePadding 属性 用 于 设置 文本 与 图 像 之 间 的 间距 。 此 实例 的 完 
整 项 日 在 MyCode\MySample021 文件 夹 中 ， 


023 ”在 TextView 中 为 文本 添加 超 链 接 


此 实例 主要 通过 设置 TextView 控件 的 autoLink 属性 为 all, 实 现 为 该 控件 的 网 址 文本 添加 超 链 
接 功 能 。 当 实例 运行 之 后 “百度 官网 : https: //www. baidu. com/” 是 一 个 TextView 控件 ,如 图 
023. 1 的 左 图 所 示 ; 单 击 其 中 的 超 链接 “https: //www. baidu. com/”, 则 将 跳 转 到 百度 官网 ,如 
图 023. 1 的 右 图 所 示 。 
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图 023. 1 
主要 代码 如 下 : 


< TextView android:layout width= " match parent" 
android:layout height = "wrap_content” 
android:textSize = "20dp” 
android:autoLink = "all" 
android:text = "百度 官网 :https://www. baidu. com/"/> 


上 面 这 段 代 码 在 MyCode\MySample366\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: autoLink 王 "all" 表 示 设 置 TextView 控件 的 所 有 文本 为 超 链 接 , 但 是 如 果 文 
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本 有 特殊 内 容 , 如 “https: //www.”, 则 将 自动 将 超 链接 定位 到 特殊 内 容 位 置 ; 如 果 文 本 的 内 容 不 是 
有 效 的 网 址 ,虽然 TextView 控件 显示 为 超 链接 ,但 是 单 击 后 会 出 现 空白 页 ,如同 在 浏览 器 的 地 址 栏 中 
输入 了 错误 的 网 址 。 此 实例 的 完整 项 目 在 MyCode\MySample366 文件 夹 中 ， 


024 在 自 定 义 View 中 实现 乖 直 滚动 文本 


此 实例 主要 通过 在 自 定 义 View 中 处 理 文本 的 行 高 和 行 数 ,实现 在 垂直 方向 上 逐 行 滚动 显示 多 行文 
本 。 当 实例 运行 之 后 ,文本 将 从 上 到 下 逐 行 滚动 显示 , 效 末 分 别 如 图 024. 1 的 左 图 和 右 图 所 示 。 
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兰州 银行 推 扫 码 取款 被 指 违规 


外 交 部 再 次 回应 韩国 部 署 萨 德 
媒体 谈 应 高 度 警 惕 蛙 人 渗透 
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主要 代码 如 下 : 


< Com.bin. luo. mysample. ScrollTextView 
android:padding = "10dp" 
android:background = " # DEFFAC" 


android:layout width= "match parent" 


android: layout_ height = "150dp" 
android:textSize = "20dp" 


外 交 部 再 次 回应 韩国 部 署 萨 德 
媒体 谈 应 高 度 警 惕 蛙 人 渗透 
印 军 称 准备 与 中 巴 两 线 作战 
高 校 禁止 外 卖 员 进 校园 

小 伙 尝 试 国庆 请 假 连 休 29 天 


图 024. 1 


android:text = "人 民 币 八 连 涨 逼近 6.5\n 
湖南 回应 教材 现 " 致 命 错误 "\n 
北京 市 暂停 共享 自行 车 投放 \n 
兰州 银行 推 扫 码 取 款 被 指 违规 \n 
外 交 部 再 次 回应 韩国 部 署 萨 德 \n 
媒体 谈 应 高 度 警惕 蛙 人 渗透 \n 
印 军 称 准备 与 中 巴 两 线 作战 \n 
高 校 禁 止 外 卖 员 进 校 园 \n 
小 伙 尝 试 国庆 请 假 连 休 29 天 " /> 


上 面 这 段 代 码 在 MyCode\MySample226\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,com. bin. luo. mysample. ScrollTextView 即 是 用 于 实现 文本 在 垂直 方 回 上 逐 行 滚动 显 
示 的 自 定义 控件 , 自 定义 控件 (类 ) ScrollTextView 的 主要 代码 如 下 : 
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public class ScrollTextView extends TextView { 
int curLines = 0,myLines = 0, perLines = 10; 
int lineHeight,myHeight, i = 0; 
boolean bScroll = true; 
Handler myHandler = new Handler() { 
Override 
public void handleMessage(Message msg) { 
super. handleMessage(msg); 
scrollTo(0, lineHeight * curLines); 
} }; 
public ScrollTextView(Context context) { 
super(context ) ; 
init( ); 
} 
public ScrollTextView(Context context, AttributeSet attrs) { 
super(context, attrs); 
init( ); 
} 
public void init() { 
new Thread(new Runnable() { 
@Override 
public void run() { 
while (true) { 
if (!bScroll) { continue; } 
if (myHeight != 0) { 
curLines = curLines + 1; 
if (curLines == (myLines - 3) * perLines) { curLines = 0; } 
Message message = myHandler.obtainMessage( ); 
message.argl = curLines; 
message. SendToTarget( ); 
try { 
if (curLines == 0) { Thread. sleep(1000);} 
else Thread. sleep(150); 
} catch (InterruptedException e) {e.printStackTrace( ); } 
} } } }). start(); 
} 
@Override 
protected void onSizeChanged(int w, int h, int oldw, int oldh) { 
super. onSizeChanged(w, h, oldw, oldh); 
curLines = 0; 


postInvalidate( ); 

lineHeight = getLineHeight() / perLines; // 获 取 行 高 
myLines = getLineCount() - 1; // 获 取 总 行 数 
myHeight = getLineCount() * lineHeight * perLines; // 获 取 总 高 度 
int height = getMeasuredHeight(); // 获 取 控 件 高 度 


i = (int) (height / getTextSize( )); 
if (myLines <= i) { bScroll = false; } 
else { bScroll = true;)} 

} 1 


上 面 这 上段 代码 在 MyCode\ MySample226 \app\src\ main\java\com\bin\luo\mysample\ 
ScrollTextView. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample226 文件 夹 中 ， 
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025 在 EditText 中 指定 输入 法 的 数 宇 软 键盘 


此 实例 主要 通过 使 用 EditText 的 setInputType() 方 法 ,实现 在 EditText 控件 获得 焦点 时 ,在 输 

人 法 的 软 键盘 上 是 显示 数字 软 键盘 ,还 是 显示 字母 软 键盘 。 当 实例 运行 之 后 , 单 击 “显示 数字 软 键盘 ” 

按钮 , 则 在 输入 法 的 软 键盘 上 显示 数字 软 键盘 ,如 图 025. 1 的 左 图 所 示 ; 单 击 “ 显 示 字 母 软 键盘 ”按钮 ， 
则 在 输入 法 的 软 键盘 上 显示 字母 软 键盘 ,如 图 025. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 
public void OnClickBtnl (View v) { // 响 应 单 击 " 显 示 数 字 软 键盘 "按钮 
myEditText. setInputType( EditorInfo. TYPE CLASS NUMBER); 


} 
public void OnClickBtn2(View v) { // 响 应 单 击 " 显 示 字 母 软 键盘 "按钮 


myEditText. setIinputType( EditorInfo. TYPE CLASS TEXT); 
} 


上 面 这 段 代码 在 MyCode\ MySample907\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample907 文件 夹 中 。 


026 ”从 止 在 EditText 中 插入 非 学 符 表 情 符 己 


此 实例 主要 通过 使 用 EditText 的 setFilters() 方 法 过 滤 输 入 的 内 容 , 实 现 禁 止 在 输入 框 中 插入 非 
字符 的 表情 符号 。 当 实例 运行 之 后 ,如 果 试 图 在 输入 框 中 输入 表情 符号 , 则 在 弹出 的 Toast 中 提示 
“此 输入 框 禁 止 输入 非 纯 文本 字符 1”, 如 图 026. 1 所 示 。 
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MySample 


Emoji 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
final EditText myEdit = (EditText) findViewById(R. id.myEdit); 
myEdit. setFilters(new InputFilter[ ]{new InputFilter(){ // 自 定义 输入 过 滤器 
(@ Override 
public CharSequence filter(CharSequence source, 
int start, int end, Spanned dest, int dstart, int dend){ 
Pattern myPattern = Pattern. compile( 
"[\ud83c\udc00 - \ud83c\udfff] |[\ud83d\udc00 - \ud83d\udfff] |[\u2600 - \u27ff]"， 
Pattern. UNICODE CASE|Pattern.CASE INSENSITIVE); 
// 对 输入 框 内 容 进行 过 滤 
Matcher myMatcher = myPattern. matcher( source); 
// 删 除 输入 框 中 非 纯 文本 字符 ,并 弹出 Toast 提示 
if (myMatcher. find( )){ 
Toast. makeText (MainActivity. this, 
"此 输入 框 禁 止 输入 非 纯 文 本 字符 !" ，Toast.LENGTH_ LONG) . show( ); 
return ""; 
} 
return null; 
133); 
六 


上 F 面 这 段 代 码 在 MyCode\ MySample639\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample639 文件 夹 中 。 


第 2 章 常用 控件 > 


027 ”使 用 AutoCompleteTextView 实现 自动 提示 


此 实例 主要 通过 使 用 AutoCompleteTextView 控件 ,实现 在 输入 框 中 输入 内 容 时 滑 出 列表 框 形 
式 的 提示 。 当 实例 运行 之 后 ,如 果 在 输入 框 中 输入 “b”, 则 在 下 面 滑 出 一 个 以 b 开头 的 列表 框 ,选择 其 
中 的 列表 项 , 则 列表 项 内 容 将 填充 到 输入 框 中 ,效果 分 别 如 图 027. 1 的 左 图 和 右 图 所 示 。 
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图 027.1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

AutoCompleteTextView myAutoCompleteTextView; 

String[ ] myArray = {"BEIJING","SHANGHAI","TIANJIN","CHONGQOING", "AKESU", 
"ANNING", "ANQING", "ANSHAN", "ANSHUN", "ANYANG", "BAICHENG", "BAISHAN", 
"BAIYIN", "BENGBU", "BAODING", "BAOJI","BAOSHAN"}; 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 

super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _main) ; 
myAutoCompleteTextView = 
(AutoCompleteTextView)findViewById(R. id.myAutoCompleteTextView); 
ArrayAdapter < String > adapter = new ArrayAdapter < String>(this, 


android. R. layout. simple dropdown item 1line, myArray); // 配 置 适配器 
myAutoCompleteTextView. setAdapter(adapter); 
myAutoCompleteTextView. setThreshold(1); // 表 示 从 第 1 个 字符 开始 显示 提示 


和 


上 F 面 这 段 代 码 在 MyCode\ MySample258\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myComplete. setThreshold(1) 表 示 从 第 1 个 字符 开始 滑 
出 列表 框 内 容 , 如 果 不 设 置 此 值 , 则 AutoCompleteTextView 控件 不 会 滑 出 下 拉 列 表 框 。 此 实例 的 完 
整 项 目 在 MyCode\MySample258 文件 夹 中 。 
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028 ”使 用 SearchView 和 ListView 实现 过 滤 输 入 


此 实例 主要 通过 综合 使 用 SearchView、ListView 等 控件 ,实现 在 输入 文字 时 过 滤 内 容 。 当 实例 
运行 之 后 ,如 果 在 SearchView 中 输入 文字 (如 “luo”), 则 在 ListView 中 显示 手机 联系 人 中 包含 输入 文 
字 ( 如 “luo”) 的 联系 人 名 称 ,如 图 028. 1 的 左 图 所 示 。 单 击 ListView 中 的 任 一 联系 人 名 称 , 如 
“luobinbin”, 则 该 联系 人 名 称 将 显示 在 SearchView 中 , 单 击 右 侧 的 右 箭头 , 则 在 弹出 的 Toast 中 显示 
SearchView 中 的 内 容 , 如 图 028. 1 的 右 图 所 示 。 
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MySample MySample 
luo| luobinbin 
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luobinbin luobinbin 


luobinbin 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
private SearchView mySearchView; 
private ListView myListView; 
private SimpleCursorAdapter myAdapter; 
private Cursor myCursor; 
static final String[ ] MYID = new String[ ]{ContactsContract. RawContacts. ID, 
ContactsContract. RawContacts. DISPLAY_NAME_ PRIMARY }; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myCursor = getContentResolver( ). query( 
ContactsContract. RawContacts. CONTENT URI, MYID, null, null, null); 
myAdapter = new SimpleCursorAdapter(this, 
android. R. layout. simple list item 1, myCursor, 
new String[ ] { ContactsContract. RawContacts. DISPLAY NAME PRIMARY }, 
new int[] { android.R. id. textl1 },0); 
myListView = (ListView) findViewById(android.R. id. list); 
myListView. setAdapter (myAdapter); 


第 2? 章 “常用 控件 《. 


myListView. setOnItemClickListener(new AdapterView. OnItemClickListener() { 
(@ Override 
public void onItemClick(AdapterView <?> parent, View view, int position, long id){ 
mySearchView. setQuery( ( (Cursor) 
myAdapter. getItem(position)).getString(1), false); 
} }); 
mySearchView = (SearchView) findViewById(R. id.mySearchView); 
mySearchView. setIconifiedBYDefault(true); 
mySearchView. onRctionViewExpanded( ); 
mySearchView. setSubmitButtonEnabled( true); 
// 监 听 搜 索 框 文本 的 改变 情况 
mySearchView. setOnQueryTextListener(new SearchView. OnQueryTextListener() { 
@ Override 
public boolean onQueryTextChangel( String queryText) { 
String mySQL = ContactsContract.RawContacts.DISPLAY NAME PRIMARY + " LIKE '%" 
+ queryText + "$, + "OR" 
+ ContactsContract.RawContacts. SORT KEY PRIMARY 
+ " LIKE '%" + queryText+ "%'"; 
myCursor = getContentResolver().query( 
ContactsContract. RawContacts. CONTENT_ URI, MYID, mySQL, null, null); 
myAdapter. swapCursor (myCursor); 
return true; 
} 
(Override 
public boolean onQueryTextSubmit(String str) {// 在 Toast 中 显示 搜索 框 文本 内 容 
Toast. makeText (MainActivity.this, str, Toast.LENGTH SHORT). show(); 
return false; 
} }); 
ke 


上 面 这 段 代 码 在 MyCode\ MySample382\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,SearchView 是 Android 原生 的 搜索 框 控 件 , 它 提供 了 一 
个 用 户 界 面 ,用 于 用 户 搜索 查询 。SearchView 默认 显示 一 个 搜索 图 标 ,点击 搜 索 图 标 则 展开 搜索 框 ， 
如 果 需 要 搜索 框 默认 展开 ,可 以 通过 setIconifiedByDefault (false) 实 现 。onQueryTextChange(String 
queryText) 重 载 方 法 在 搜索 框 内 容 发 生变 化 时 啊 应 ,此 实例 在 此 实现 根据 搜索 框 内 容 过 滤 手 机 联系 
人 。onQueryTextSubmit(String str) 重 载 方法 在 单 击 了 右 侧 的 右 箭 头 时 啊 应 ,此 实例 在 此 实现 在 
Toast 中 显示 搜索 框 内 容 。 此 外 , 读 取 手机 联系 人 信息 需要 相关 的 权限 。 因 此 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 一 "android. permission. READ 


CONTACTS" /> 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample382 文件 夹 中 ，。 


029 在 EditText 右 端 设置 输入 提示 内 容 和 图 标 


此 实例 主要 通过 使 用 EditText 的 setError() 方 法 ,实现 在 EditText 的 右 端 设置 提示 内 容 和 图 
标 。 当 实例 运行 之 后 ,在 EditText 的 右 端 将 显示 灯泡 图 标 和 提示 文字 “必须 填写 ”, 如 图 029. 1 的 左 图 
所 示 。 如 果 不 设置 自己 的 图 标 ,Android 将 显示 一 个 默认 的 感叹 号 图 标 ,， 如 图 029. 1 的 右 图 所 示 。 
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图 029.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
EditText myEditText; 
(@ Override 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity_main); 
myEditText = (EditText) findViewById(R. id. myEditText) ; 
Drawable myDrawable = getResources().getDrawable(R.mipmap.myimagel ); 
myDrawable. setBounds(0, 0,72,72); 
myEditText. setError(" 必须 填写 " myDrawable); 
//myEditText. setError(" * * *"); 
| 


上 面 这 段 代 码 在 MyCode\ MySample629\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myEditText. setError(" 必 须 填写 ", myDrawable) 用 于 在 
EditText 上 设置 错误 提示 内 容 及 图 标 , 其 实 TextView 也 可 以 如 此 设置 错误 提示 内 容 , 不 过 它 要 获取 
焦点 (TextView. requestFocus()) 才 能 显示 错误 信息 ,而 EditText 不 用 获取 焦点 是 因为 自己 能 够 抢 到 
焦点 。 此 实例 的 完整 项 目 在 MyCode\MySample629 文件 夹 中 。 


030 ”通过 目 定 义 Shape 创建 不 同 的 贺 角 按钮 


此 实例 主要 通过 在 选择 器 (selector) 文 件 中 配置 按钮 在 按 下 和 抬 起 时 的 Shape, 从 而 创建 不 同 的 
圆 角 按 钮 。 当 实例 运行 之 后 ,所 有 按钮 的 右上 角 和 右 下 角 都 是 圆 角 ,而 左上 角 和 左下 角 都 是 直角 ; 单 
击 第 一 个 按钮 ,效果 如 图 030. 1 的 左 图 所 示 ; 单 击 第 三 个 按钮 ,效果 如 图 030. 1 的 右 图 所 示 ; 单 击 其 
他 按钮 会 实现 类 似 的 功能 。 


MySample 


电视 剧 频道 
微 电 影 频道 
真人 秀 频道 


MySample 


电影 频道 


电视 剧 频道 


图 030. 1 


主要 代码 如 下 : 


<LinearLayout android:orientation = "vertical" 


android: layout width = "wrap_content" 


android:layout height = "wrap content"> 


<Button android:layout margin= "5dp" 


android: background = " (Q drawable/buttonselector" 


android:layout width= "200dp" 


android:layout height = "wrap_content" 


android:textSize = "20dp" 
android:text = "电影 频道 "/> 
< Button android:layout margin= "5dp" 


android: background = " (odrawable/buttonselector" 


android:layout width= "200dp" 


android:layout height = "wrap_content" 


android:textSize = "20dp" 


android:text = "电视 剧 频 道 "/> 
<Button android:layout margin = "5dp" 


android: background = " (Wdrawable/buttonselector" 


android:layout width= "200dp" 


android:layout height = "wrap_content" 


android:textSize = "20dp" 
android:text =" 微 电影 频道 "/> 
<Button android:layout margin = "5dp" 


android: background = " (Wdrawable/buttonselector" 


android:layout width= "200dp" 


android:layout height = "wrap_content" 


android:textSize = "20dp" 
android:text = "真人 秀 频 道 "/> 


</LinearLayout > 
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上 面 这 段 代 码 在 MyCode\MySample352\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android:background 二 "(@drawable/buttonselector" 表 示 按 钮 采用 buttonselector 资源 
进行 背景 设置 。buttonselector 是 一 个 XML 文件 ,该 文件 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< selector xmlns:android = "http://schemas.android. com/apk/res/android"> 
< item android: drawable = " (Wdrawable/buttonpressed" 
android: state pressed = "true"/> 
< item android: drawable = " (Wdrawable/buttonnormal" 
android: state pressed = "false" /> 
</selector > 


上 面 这 段 代 码 在 MyCode\MySample352\app\src\main\res\drawable\buttonselector. xml 文件 
中 。 在 这 段 代 码 中 ,< item android: drawable 王 "@ drawable/buttonpressed" android: state_pressed 
一 "true" /> 表示 当 按 钮 处 于 按 下 状态 时 ,设置 buttonpressed 资源 。< item android: drawable 王 "@ 
drawable/buttonnormal"” android: state _ pressed 一" false"/ > 表示 当 按 钮 处 于 正常 状态 时 ,设置 
buttonnormal 资源 。buttonpressed 资源 和 buttonnormal 资源 都 是 XML 文件 ,buttonpressed 资源 的 
主要 内 容 如 下 : 


<?xml] version = "1.0" encoding = "utf 一 8"?> 
<! -- 按钮 按 下 时 候 的 背景 -一 > 
< shape xmlns:android = "http://schemas.android. com/apk/res/android"> 
<! -- 按钮 右上 角 和 右 下 角 的 圆 角 半 径 值 --> 
< Corners android:topRightRadius = " 30dp" 
android: bottomRightRadius = "30dp" /> 
<! -- 按钮 的 填充 色 --> 
< solid android: color = " #CDOOCD" /> 
</shape> 


上 面 这 段 代 码 在 MyCode\MySample352\app\src\main\res\drawable\buttonpressed. xml 文件 
中 。 在 这 段 代 码 中 ,android: topRightRadius 表示 设置 按钮 右上 角 的 圆 角 半径 ,android: 
bottomRightRadius 表示 设置 按钮 右 下 角 的 圆 角 半径 。topLeftRadius 表示 设置 按钮 左上 角 的 圆 角 半 
径 ,bottomLeftRadius 表示 设置 按钮 左下 角 的 圆 角 半径 ,radius 表示 设置 按钮 的 4 个 角 的 圆 角 半径 。 
如 果 未 设置 这 些 属性 , 则 按钮 的 4 个 角 将 显示 为 直角 。buttonnormal 资源 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<! -- 按钮 正常 时 候 的 背景 --> 
< shape xmlns:android = "http://schemas. android. com/apk/res/android"> 
<! -- 按钮 的 右上 角 和 右 下 角 圆 角 半 径 --> 
< corners android:topRightRadius = " 30dp" 
android: bottomRightRadius ="30dp" /> 
<! -- 按钮 的 填充 色 一 -> 
< solid android: color =" #98F5FF" /> 
<! -一 按钮 边框 的 宽度 ,每 段 虚线 的 长 度 和 两 段 虚线 之 间 的 颜色 -一 > 
| 二 二 六 
<! —— android:width= "ldp" ——> 
<! —— android:dashWidth = "8dp" ——> 
<! —— android:dashGap = "4dp" -一 > 
<! —— android:color =" 井 4eb621" /> -一 > 
</shape > 
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上 和 面 这 段 代 码 在 MyCode\MySample352\app\src\main\res\drawable\buttonnormal. xml 文件 


中 。 此 实例 的 完整 项 目 在 MyCode\MySample35 


031 


感 按钮 。 当 实例 运行 之 后 
在 弹出 的 Toast 中 显示 “同意 


2 文件 夹 中 。 


设置 背景 图 像 创 建立 体 的 质感 按钮 


此 实例 主要 通过 设置 Button 控件 的 background 属性 为 指定 的 按钮 图 像 ,创建 具有 立体 效果 的 质 


中 显示 “此 协议 待 修正 ”, 如 图 031. 1 的 右 图 所 示 。 


单方 保生 


后 “确认 ”和 ”取消 ”按钮 的 背景 是 使 用 图 像 设 置 的 背景 , 单 击 “ 确 认 ” 按 钮 , 则 
此 协议 ”, 如 图 031. 1 的 左 图 所 示 ; 单 击 “取消 ?按钮 , 则 在 弹出 的 Toast 


Q bm VG 中 


MySample 


同意 此 协议 


方 单方 面 负 有 保密 愉 
议 。 


主要 代码 如 下 : 


确认 


方 单方 面 负 有 保本 
iv 癸 


确认 


图 031. 1 


<LinearLayout androlid:layout_width = "match parent" 


android:layout height = "wrap_content” 


android:gravity = "center" 
android:orientation = "horizontal"> 


< Button android: 
android: 
android: 
android: 
android: 
android: 
android: 
android: 
< Button android: 
android: 
android: 


layout width= "180dp" 
layout height = "100dp" 
background = "(@drawable/mypng" 
onClick = "onClickmyBtnl1" 
paddingBottom = "5dp" 
text = "确认 " 

textColor = "#fff" 
textSize = "22dp" /> 

id = "(@ + id/myBtn2" 
layout width= "180dp" 
layout height = "100dp" 
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android: background = " (Qdrawable/mypngy" 

android: onClick = "onClickmyBtn2" 

android:paddingBottom = "5dp" 

android:text = "取消 " 

android:textColor = "# fff" 

android:textSize = "22dp"/> 
</LinearLayout > 


上 面 这 段 代码 在 MyCode\MySample048\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: background 王 "(@ drawable/mypng" 表 示 采 用 drawable 中 的 mypng 图 像 资 
源 设 置 按钮 的 背景 。android: onClick 二 "onClickmyBtn2" 用 于 将 按钮 单 击 事件 与 onClickmyBtn2 方 
法 关联 起 来 。onClickmyBtn2 方法 的 主要 代码 如 下 : 


// 响 应 单 击 "取消 "按钮 
public void onClickmyBtn2(View v) { 
Toast myToast = Toast.makeText(getApplicationContext(), 
"此 协议 待 修正 " ，Toast.LENGTH LONG) ; 
myToast. setGravity(Gravity. RIGHT | Gravity. TOP, 360, 870); 
myToast. show( ); 


} 


上 面 这 段 代 码 在 MyCode\ MySample048\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,onClickmyBtn2 方法 的 主要 作用 就 是 在 Toast 中 显示 提示 
信息 。 此 实例 的 完整 项 目 在 MyCode\MySample048 文件 夹 中 。 


032 ”使 用 FloatingActionButton 创建 基 学 按钮 
此 实例 主要 通过 使 用 FloatingActionButton 控件 ,实现 悬浮 按钮 的 效果 。 当 实例 运行 之 后 ,将 在 


屏幕 中 心 显示 一 个 悬浮 按钮 ,如 图 032. 1 的 左 图 所 示 ; 单 击 该 悬浮 按钮 , 则 在 下 面 弹 出 一 个 Toast, 如 
图 032. 1 的 右 图 所 示 。 


刚才 单 击 了 FloatingActionButton 按 钮 


图 032. 1 
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主要 代码 如 下 : 


< android. support. design. widget. FloatingActionButton 
android: layout _ width = "wrap_content" 
android: layout height = "wrap content" 
android: onClick = "onClickmyBtn1" 
android: src = " (Omipmap/myimage" 
android: layout_ centerInParent = "truen 
app: fabSize = "normal" /> 


上 面 这 段 代 码 在 MyCode\MySample374\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: lavyout_centerInParent 王 "true" 表 示 悬 浮 按钮 位 于 屏幕 正中 ,如 果 需 要 悬浮 按 
钮 位 于 右 下 和 角 , 则 应 该 设置 android: layout _ alignParentRight 二 "true" 和 android: layout _ 
alignParentBottom 一 “true" 。 此 外 ,FloatingActionButton 可 以 通过 fabSize 属性 设置 大 小 ,正常 size 
是 normal, 还 有 一 个 更 小 的 模式 为 mini。 需 要 说 明 的 是 ,在 Android Studio 中 使 用 FloatingActionButton 
控件 需要 在 gradle 中 引入 compile 'com. android. support: design: 24. 2.0' 依 赖 项 ,24 是 当前 开发 环 
境 的 SDK 版 本 号 。 此 实例 的 完整 项 目 在 MyCode\MySample374 文件 夹 中 。 


033 ”以 全 屏 效果 显示 在 ImageView 中 的 图 像 


此 实例 主要 通过 设置 ImageView 控件 的 android: scaleType 属性 为 fitXY ,从 而 实现 以 全 屏 效 果 
显示 图 像 。 当 实例 运行 之 后 ,图像 全 屏 显 示 的 效果 如 图 033. 1 所 示 。 


033.1 


《G2》 anaroia 引 本 应 用 200 实战 篇 


主要 代码 如 下 : 


< ImageView android:]layout width= "match _ parent” 
android:layout height = "match parent" 
android: scaleType = "fitXY” 
android: src = "(@mipmap/myimage" /> 


上 了 面 这 段 代 码 在 MyCode\MySample059\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 上段 代码 中 ,android: src 二 "(@@mipmap/myimage" 用 于 设置 ImageView 控件 显示 的 图 像 ,myimage 
是 图 像 资源 , 它 是 以 资源 的 形式 将 同名 文件 添加 到 此 实例 项 目的 “app\src\main\res\mipmap” 节 点 
(目录 ) 下 。android: scaleType= "fitXY" 表 示 将 图 像 无 条 件 填 满 控 件 的 窗口 。 默 认 情 况 下 ,Android 
应 用 会 显示 标题 栏 和 通知 栏 等 装饰 栏 ,全 屏 显 示 通 稼 应 该 隐藏 这 些 装饰 栏 ,这 可 以 通过 Java 代码 来 实 
现 , 主 要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
// 全 屏 设置 ,隐藏 窗口 所 有 装饰 
getWindow( ). setFlags(WindowManager. LayoutParams. FLAG FULLSCREEN, 
WindowManager. LayoutParams. FLAG FULLSCREEN); 
// 标 题 是 属于 View 的 ,所 以 窗口 所 有 的 装饰 部 分 被 隐藏 后 标题 依然 有 效 , 须 隐藏 
requestWindowFeature(Window. FEATURE NO _ TITLE ) ; 
setContentView(R. layout. activity main); 
1 


上 面 这 段 代 码 在 MyCode\ MySample059\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , getWindow (). setFlags (WindowManager. LayoutParams. 
FLAG FULLSCREEN, WindowManager. LayoutParams. FLAG_ FULLSCREEN) 和 requestWindowFeature 
(Window. FEATURE_NO_TITLE) 代 码 必 须 在 setContentView(R. layout. activity_main) 之 前 调用 ， 
否则 会 报错 。 此 实例 的 完整 项 目 在 MyCode\MySample059 文件 夹 中 。 


034 ”在 自 定义 ImageView 中 显示 圆 形 图 像 


此 实例 主要 通过 以 ImageView 类 为 基 类 创建 自 定义 控件 CircleImageView ,实现 在 ImageView 
控件 中 以 圆 形 的 风格 显示 图 像 。 当 实例 运行 之 后 , 圆 形 图 像 的 显示 效果 如 图 034. 1 所 示 。 
主要 代码 如 下 : 


< com. bin. luo. mysample. CircleImageView 
android:layout_gravity = "center" 
android:layout margin = "35dp" 
android: id= "@ + id/myView" 
android:layout width= "wrap_content" 
android:layout_ height = "wrap_content” 
android: src = "(@mipmap/myimage" /> 


上 面 这 段 代 码 在 MyCode\MySample282\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 人 码 中 ,com. bin. luo. mysample. CircleImageView 即 是 用 于 实现 图 像 以 圆 形 的 风格 显示 的 自 


Q 


MySample 


图 034.1 
定义 控件 ,CircleImageView 类 的 主要 代码 如 下 : 


public class CircleImageView extends ImageView { 
public CircleImageView(Context context) { super(context); } 
public CircleImageView(Context context, AttributeSet attrs) { 
super (context, attrs); 
} 
public CircleImageView(Context context, AttributeSet attrs, int defStyle) { 
super(context, attrs, defStyle); 
} 
(Override 
protected void onDraw(Canvas myCanvas) { 
Drawable myDrawable = getDrawablel( ) ; 
if (myDrawable == null) { return; } 
if (getWidth() == 0 || getHeight() == 0) { return; } 
Bitmap myBitmap = ((BitmapDrawable) myDrawable). getBitmap( ) ; 
Bitmap oldBmp = myBitmap. copy(Bitmap. Config. ARGB 8888, true); 
int myRadius = getWidth!( ) ; 
Bitmap CircleBmp = getCroppedBitmap(oldBmp, myRadius); 
myCanvas. drawBitmap(CircleBmp, 0, 0, null); 
} 
public static Bitmap getCroppedBitmap(Bitmap oldBmp, int myRadius) { 
Bitmap myBmp; 
if (oldBmp.getWidth() != myRadius | | oldBmp. getHeight() != myRadius){ 
myBmp = Bitmap.createScaledBitmap(oldBmp, myRadius, myRadius, false); 
} else{ myBmp = oldBmp; } 
Bitmap newBmp = Bitmap. createBitmap(myBmp. getWidth( ), 
myBmp. getHeight(), Bitmap. Config. ARGB 8888 ) ; 
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Canvas myCanvas = new Canvas(newBmp); 
final Paint myPaint = new Paint( ); 
final Rect myRect = new Rect(0, 0, myBmp.getWidth(), myBmp. getHeight()); 
myPaint. setAntiAlias(true); 
myCanvas. drawCircle(myBmp. getWidth() / 2, 

myBmp. getHeight() / 2, myBmp. getWidth() / 2, myPaint); 
myPaint. setXfermode( new PorterDuffXfermodel( PorterDuff. Mode. SRC IN)); 


myCanvas. drawBitmap( myBmp, myRect, myRect, myPaint); 
return newBmp; 


， 


上 面 这 段 代 码 在 MyCode\ MySample282\app\src\main\java\com\bin\luo\mysample\ 
CircleImageView. java 文件 中 。 在 这 段 代 码 中 ,myBitmap 二 ((BitmapDrawable) myDrawable). 
getBitmap() 用 于 获取 CircleImageView 已 加 载 图 像 。newBmp 二 Bitmap. createBitmap (myBmp. 
getWidth() ,Bmp. getHeight() ，Bitmap. Config. ARGB 8888) 用 于 根据 图 像 的 大 小 创建 空白 的 新 位 
图 。myCanvas 二 new Canvas(newBmp) 用 于 根据 空白 的 新 位 图 创建 新 画布 。myCanvas. drawCircle 
(myBmp. getWidth()/2,myBmp. getHeight()/2, myBmop. getWidth() / 2, myPaint) 用 于 在 画布 上 
绘制 与 图 像 大 小 相同 的 圆 形 。myPaint. setXfermode(new PorterDuffXfermode (PorterDuff. Mode. 
SRC_IN) ) 用 于 控制 圆 形 如 何 与 将 要 绘制 的 图 像 进 行 交 互 (裁剪 )。myCanvas. drawBitmap (myBmp， 
myRect，myRect，myPaint) 用 于 根据 指定 的 PorterDuff. Mode. SRC_IN 模式 在 圆 形 中 绘制 图 像 。 
此 实例 的 完整 项 目 在 MyCode\MySample282 文件 夹 中 。 


035 ”使 用 单 指 滑动 拖 蝶 ImageView 的 图 像 


此 实例 主要 通过 在 ImageView 的 setOnTouchListener() 方 法 中 监听 手指 滑动 的 位 置 变 化 ,实现 
拖 电 ImageView 控件 的 图 像 。 当 实例 运行 之 后 ,使 用 手指 按 住 图 像 , 然 后 在 屏幕 上 滑动 , 则 图 像 将 跟 
随 滑动 ,效果 分 别 如 图 035. 1 的 左 图 和 右 图 所 示 。 

和 D 和 


Ed 6:35 


MySample MySample 


图 035.1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
private PointF myStartPoint = new PointF(); 
private Matrix myMatrix = new Matrix(); 


private Matrix myCurrentMatrix = new Matrix(); 


private int MODE = 0; // 用 于 判断 当前 手势 是 否 为 拖 忠 操作 
private static final int DRAG = 1; // 拖 电 操 作 标 志 
(Override 


protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
final ImageView myImageView = (ImageView) findViewById(R. id.myImageView); 
myImageView. setScaleType( ImageView. ScaleType. MATRIX); 
myImageView. setOnTouchListener(new View. OnTouchListener() { 
(Override 
public boolean onTouch(View v, MotionEvent event) { 
Switch (event. getAction() & MotionEvent. ACTION MASK) { 
case MotionEvent. ACTION DOWN: 
= DRAG; 
myCurrentMatrix. set(myImageView. getImageMatrix( ) ); 
myStartPoint. set(event.getX(), event.getY());  // 记 录 起 始 位 置 


break; 

case MotionEvent. ACTION MOVE: 

if (MODE == DRAG) { // 拖 电 发 生 
float dx = event.getX() - myStartPoint.x; //x 轴 移动 距离 


float dy = event.getY() - myStartPoint.y; //Y 轴 移动 距离 
myMatrix. set(myCurrentMatrix); 
myMatrix. postTranslate(dx, dy); // 和 矩阵 位 移 操作 


break; 


myImageView. setImageMatrix(myMatrix); 
return true; 
} }); 
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上 面 这 段 代 码 在 MyCode\ MySample676\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImageView. setImageMatrix(myMatrix) 用 于 根据 
myMatrix 参数 平移 (滑动 ) 图 像 。 myMatrix. postTranslate(dx, dy) 则 用 于 根据 参数 定制 myMatrix, 
dx 参数 表示 在 工 轴 上 的 平移 值 ,dy 参数 表示 在 y 轴 上 的 平移 值 。 注 意 , 在 ImageView 控件 中 使 用 
Matrix 平移 图 像 时 ,通常 应 用 调用 ImageView 的 setScaleType (ImageView. ScaleType. MATRIX) 
方法 。 此 实例 的 完整 项 目 在 MyCode\MySample676 文件 夹 中 。 
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036 ”使 用 Gallery 实现 滑动 浏览 多 幅 图 像 


此 实例 主要 通过 使 用 Gallery 控件 和 ImageView 控件 ,实现 以 手指 左右 滑动 的 方式 浏览 画廊 的 
多 幅 图 像 的 功能 。 当 实例 运行 之 后 ,如 果 手 指 按 住 图 像 回 左 滑 动 , 则 滑 出 右 侧 隐 藏 的 图 像 ; 如 果 手 指 
按 住 图 像 回 右 滑 动 , 则 滑 出 左 侧 隐藏 的 图 像 ,效果 分 别 如 图 036. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


图 036. 1 
主要 代码 如 下 : 


< Gallery android:layout width= "match _ parent"” 
android:layout height = "wrap_content” 
android: id = "(@ + id/myGallery" /> 


上 和 面 这 段 代 码 在 MyCode\MySample058\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,Gallery 控件 用 于 显示 夯 亡 图 像 , 它 主 要 通过 自 定义 的 ImageAdapter 管理 多 幅 图 像 。 
自 定 义 ImageAdapter 的 主要 代码 如 下 : 


public class MainActivity extends Activity { 
(@ Override 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
Gallery myGallery = (Gallery) findViewById(R. id.myGallery); 
// 添 加 一 个 ImageAdapter 并 配置 给 Gallery 
myGallery. setAdapter(new ImageAdapter(MainActivity. this)); 
} 
public class ImageAdapter extends BaseAdapter { 
private Context myContext; 
private int[ ] myImages = {R.drawable.mypngl, R.drawable.mypng2, 
R. drawable. mypng3, R.drawable.mypng4, R. drawable.mypng5, 
R. drawable.mypng6，R. drawable. mypng7，R. drawable. mypng8 } ; 
public ImageAdapter(Context c) { this.myContext = cj } 
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// 获 取 已 定义 的 图 像 总 数量 

public int getCount() { return myImages. length; } 

// 获 取 指 定位 置 的 图 像 

public Object getItem(int position) { return position; } 

// 获 取 指 定位 置 的 图 像 ID 

public long getItemId(int position) { return position; } 

public View getView( int position, View convertView, ViewGroup parent) { 


ImageView myView = new ImageView(myContext); // 创 建 ImageView 对 象 
myView. setImageResource(myImages[ position]); // 设 置 ImageView 的 图 像 
myView. setScaleType( ImageView. ScaleType. FIT XY); // 重 新 设置 图 像 的 宽 高 
myView. setPadding(25, 25, 25, 25); // 重 新 设置 图 像 的 内 边 距 
myView. setLayoutParams( 

new Gallery. LayoutParams(220, 220)); // 重 新 设置 Layout 的 宽 高 


return myView; 


} 
// 根 据 距离 中 央 的 位 移 量 ,利用 getScale() 返 回 大 小 
public float getScale( boolean focused, int offset) { 
return Math. max(0, 1.0f / (float) Math. pow(2, Math.abs(offset))); 


} }} 


上 面 这 上段 代码 在 MyCode\ MySample058\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 , R. drawable. mypngl，R. drawable. mypng2，R. 
drawable. mypng3 ,人 . drawable. mypng4, R. drawable. mypng5, R. drawable. mypng6, R. drawable. 
mypng7，R. drawable. mypng8 是 画廊 的 图 像 资 源 。 它 是 以 资源 的 形式 添加 到 此 实例 项 目的 “app\src 
\main\res\drawable” 节 点 下 。 此 实例 的 完整 项 目 在 MyCode\ MySample058 文件 夹 中 。 


037 ”使 用 SwipeRefreshLayout 切换 图 像 


此 实例 主要 通过 使 用 SwipeRefreshLayout, 实 现 切 换 图 像 。 当 实例 运行 之 后 , 按 住 屏幕 下 滑 , 则 
将 出 现 一 个 转圈 的 动画 表示 正在 刷新 ,1 秒 之 后 转圈 动画 停止 ,然后 显示 一 幅 不 同 的 图 像 ( 电 影 海报 )， 
如 图 037. 1 的 左 图 所 示 。 再 次 按 住 屏 幕 下 滑 ,也 将 出 现 一 个 转圈 的 动画 表示 正在 刷新 ,1 秒 之 后 转圈 
动画 停止 ,然后 又 显示 一 幅 不 同 的 图 像 ( 电 影 海报 ) ,如 图 037. 1 的 右 图 所 示 。 
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public class MainActivity extends AppCompatActivity { 

ImageView myImageView; 

Boolean bChecked = true; 

(Override 

protected void onCreate( (@Nullable Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
final SwipeRefreshLayout mySwipeRefreshLayout = (SwipeRefreshLayout) 

findViewById(R. id. mySwipeRefreshLayout) ; 

myImageView = (ImageView) findViewById(R. id. myImageView) ; 
mySwipeRefreshLayout. setOnRefreshListener( 


new SwipeRefreshLayout. OnRefreshListener() { // 下 拉 时 执行 更 新 操作 
(@ Override 


public void onRefresh() { 
mySwipeRefreshLayout. setRefreshing(true); 
(new Handler()).postDelayed(new Runnable() { 
(Override 
public void run() { 
mySwipeRefreshLayout. setRefreshing(false); 


if (bChecked) { // 显 示 第 一 幅 图 像 

myImageView. setImageDrawable( getDrawable(R. mipmap. myimage!l ) ); 

} else { // 显 示 第 二 幅 图 像 

myImageView. setImageDrawable( getDrawable(R. mipmap. myimage2 ) ); 

} 

bChecked = !bChecked; 

} }, 1000); // 更 新 间隔 时 间 1 秒 
Fj 


上 面 这 上段 代码 在 MyCode\ MySample645\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 Android 中 ,SwipeRefrshLayout 控件 主要 用 于 实现 下 拉 刷 新 的 动态 效 
果 ,基本 方法 说 明 如 下 。 

(1) setOnRefreshListener(OnRefreshListener) ,该 方法 用 于 添加 下 拉 刷 新 监听 顺 。 

(2) setRefreshing(boolean) ,该 方法 用 于 显示 或 者 隐藏 刷新 进度 条 。 

(3) isRefreshing() ,该 方法 检查 是 否 处 于 刷新 状态 。 

(4) setColorSchemeResources() ,该 方法 设置 进度 条 的 颜色 主题 ,最 多 设置 4 种 。 

大 多 数 情况 下 ,使 用 SwipeRefrshLayout 控件 均 需 要 处 理 setOnRefreshListener() 方 法 ,在 其 中 
实现 刷新 的 目的 (如 此 实例 的 切换 图 像 ) 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 
compile ' com. android. support: design: 25. 0. 1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample645 文件 夹 中 。 


038 ”使 用 AdapterViewFlipper 自动 播放 图 像 


此 实例 主要 通过 使 用 AdapterViewFlipper 控件 ,实现 日 动 播放 在 图 库 中 的 多 幅 图 像 。 当 实例 运 
行 之 后 , 单 击 * 上 一 幅 图 像 ” 按 钮 , 则 将 显示 当前 图 像 的 上 一 幅 图 像 ,如 图 038. 1 的 左 图 所 示 。 单 击 “ 下 
一 幅 图 人像” 按钮 , 则 将 显示 当前 图 像 的 下 一 幅 图 像 ,如 图 038. 1 的 右 图 所 示 。 单 击 “ 自 动 播放 ”按钮 , 则 
将 依次 循环 显示 在 图 库 中 的 多 幅 图 像 ; 如 果 已 经 是 图 库 的 最 后 一 幅 图 像 , 则 下 一 幅 图 像 即 是 图 库 的 第 
一 幅 图 像 。 
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上 一 幅 图 像 ” 下 一 幅 图 像 ” 自动 播放 中 上 一 幅 图 像 ”下 一 幅 图 像 ”自动 播放 


图 038.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
int[ ] myImages = new int[ ]{R.mipmap.myimagel，R.mipmap.myimage2， 
R. mipmap. myimage3, R. mipmap. myimage4,R.mipmap. myimage5, 
R. mipmap. myimage6, R. mipmap. myimage7 }; 
AdapterViewFlipper myAdapterViewFlipper; 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myAdapterViewFlipper = 
(AdapterViewFlipper)findViewById(R. id.myAdapterViewF]l ipper); 
// 创 建 BaseAdapter 对 象 ,该 对 象 负责 提供 图 库 的 列表 项 
BaseAdapter myBaseAdapter = new BaseAdapter( ){ 
(Override 
public int getCount() { return myImages. length; } 
(Override 
public Object getItem(int position) { return position; } 
(QOverride 
public long getItemId(int position) { return position; } 
// 该 方法 返回 的 View 代表 了 每 个 列表 项 
(Override 
public View getView(int position, View convertView, ViewGroup parent) { 
ImageView myImageView = new ImageView(MainActivity. this); 
myImageView. setImageResource(myImages[ position]); 
// 设 置 缩放 类 型 
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myImageView. setScaleTYpe(ImageView. ScaleType. FIT XY); 
// 为 ImageView 设置 布局 参数 
myImageView. setLayoutParams(new ViewGroup. LayoutParams( 
ViewGroup. LayoutParams. MATCH PARENT, 
ViewGroup. LayoutParams. MATCH_ PARENT) ); 
return myImageView; 
} 
}; 
myAdapterViewFlipper. setAdapter(myBaseAdapter); 
} 
// 响 应 单 击 " 上 一 幅 图 像 "按钮 
public void onClickmyBtnl(View v){ 
myAdapterViewFlipper. showPrevious( ); 
myAdapterViewFlipper. stopFlipping( ); 
} 
// 响 应 单 击 " 下 一 幅 图 像 "按钮 
public void onClickmyBtn2(View v){ 
myAdapterViewFlipper. showNext( ); 
myAdapterViewFlipper. stopFlipping( ); 


} 
// 响 应 单 击 " 自 动 播放 "按钮 
public void onClickmyBtn3(View v){ 


myAdapterViewFlipper. startFlipping( ); 
可 


上 F 面 这 段 代 码 在 MyCode\ MySample384\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,AdapterViewFlipper 控件 可 以 显示 Adapter 提供 的 View 
控件 ,但 是 每 次 只 能 显示 一 个 ,可 以 通过 该 控件 的 showPrevious() 和 showNext() 方 法 实现 显示 上 一 
个 或 下 一 个 View 控件 ; 除 此 之 外 , 它 还 可 以 通过 startFlipping() 方 法 实现 自动 播放 控件 ,也 可 以 通过 
stopFlipping() 方 法 停止 自动 播放 。AdapterViewFlipper 控件 常用 的 属性 如 下 。 

(1) android: animateFirstView 属性 ,该 属性 用 于 设置 在 显示 该 控件 的 第 一 个 View 时 是 否 使 用 
动画 。 

(2) android: inAnimation 属性 ,该 属性 用 于 设置 控件 在 显示 时 使 用 的 动画 。 

(3) android: loopViews 属性 ,该 属性 用 于 设置 循环 到 最 后 一 个 控件 后 是 否 自 动 * 转 头 ” 到 第 一 个 
控件 。 

(4) androld: outAnimation 属性 ,该 属性 用 于 设置 控件 在 隐藏 时 使 用 的 动画 。 

(5) android: autoStart 属性 ,该 属性 用 于 设置 该 控件 是 否 是 自动 播放 动画 ，。 

(6) android: flipInterval 属性 ,该 属性 用 于 设置 自动 播放 的 时 间 间 隅 。 

此 实例 的 完整 项 目 在 MyCode\MySample384 文件 夹 中 ，。 


039 ”使 用 两 幅 图 像 定制 ToggleButton 开关 状态 


此 实例 主要 通过 在 选择 峰 (selector) 文 件 中 配置 ToggleButton 的 state_checked 属性 的 不 同 
(true/false) 值 对 应 的 图 像 , 从 而 实现 定制 ToggleButton 的 开关 状态 。 当 实例 运行 之 后 ， 
ToggleButton 处 于 关闭 状态 ,没有 显示 背景 图 像 ,如 图 039. 1 的 左 图 所 示 ; 单 击 ToggleButton, 则 
ToggleButton 处 于 打开 状态 ,立即 显示 背景 图 像 , 如 图 039. 1 右 图 所 示 。 
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主要 代码 如 下 : 


<LinearLayout android:layout width= "match parent" 
android:layout height = "match parent" 
android:gravity = "center" 
android: id = "(@ + id/myLinearLayout" 
android:orientation = "horizontal"> 
<TextView android: id = "(@ + id/myTextView" 
android:layout width = "wrap_ content" 
android:layout height = "wrap_content” 
android:text = "已 关闭 背景 图 像 模式 " 
android:textColor = "(Qandroid:color/black" 
android:textSize = "20. 0dp"/> 
<ToggleButton android: id = "(0@ + id/myToggleButton" 
android: layout width = "wrap content" 
android: layout height = "wrap content" 
android: background = " (Wdrawable/myselector" 
android: checked = "false" 
android:text ="" 
android: textOff ="" 
android:textOn="" /> 
</LinearLayout > 


上 面 这 段 代码 在 MyCode\MySample053\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: background 王 "(@ drawable/myselector" 表 示 ToggleButton 采用 myselector 
资源 进行 背景 配置 。myselector 是 一 个 XML 文件 ,该 文件 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< Selector xmlns:android = "http://schemas. android. com/apk/res/android"> 

< item android: drawable = " (Wdrawable/toggleopen" android: state checked = "true" /> 

< item android: drawable = " (Wdrawable/toggleclose" android: state checked = " false" /> 
</ selector > 


S Android 炫 酷 应 用 300 例 。 实战 篇 


上 面 这 段 代 码 在 MyCode\MySample053\app\src\main\res\drawable\myselector. xml 文件 中 。 
在 这 段 代 码 中 ,< item android: drawable = 二"(@ drawable/toggleopen" android: state _checked 一" 
true" /> 表示 当 ToggleButton 的 state_checked 属性 值 为 true 时 ,设置 toggleopen 资源 。< item 
android: drawable 一" (@ drawable/toggleclose” android: state _ checked 一" false"/ > 表示 当 
ToggleButton 的 state_checked 属性 值 为 false 时 ,设置 toggleclose 资源 。 

ToggleButton 的 CheckedChange 事件 啊 应 方法 则 通过 Java 代码 实现 ,主要 代码 如 下 : 


public class MainActivity extends Activity { 
private ToggleButton myToggleButton; 
private TextView myTextView; 
private LinearLayout myLinearLayout; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myLinearLayout = (LinearLayout) findViewById(R. id. myLinearLayout) ; 
myTextView = (TextView) findViewById(R. id. myTextView) ; 
myToggleButton = (ToggleButton) findViewById(R. id. myToggleButton) ; 
// 响 应 单 击 ToggleButton 
myToggleButton. setOnCheckedChangeListener( 
new CompoundButton. OnCheckedChangeListener() { 
public void onCheckedChanged( CompoundButton buttonView, 
boolean isChecked) { 
if (isChecked) { 
myTextView. setText(" 已 开启 背景 图 像 模 式 " ); 
myLinearLayout. setBackground( 
getResources( ) .getDrawable(R. mipmap. mybackground) ) ; 
} else { 
myTextView. setText(" 已 关闭 背景 图 像 模 式 " ); 
myLinearLayout. setBackground(null); 
3} 
} 


上 面 这 段 代 码 在 MyCode\ MySample053\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 , myLinearLayout. setBackground (null) 表示 取消 屏幕 
(LinearLayout) 背景 图 像 。myLinearLayout. setBackground (getResources ( ). getDrawable (R. 
mipmap. mybackground) ) 表 示 使 用 mybackground 资源 设置 屏幕 的 背景 图 像 。 此 实例 的 完整 项 目 在 
MyCode\MySample053 文件 夹 中 。 


040 使 用 GridView 创建 网 格 显示 多 幅 图 像 


此 实例 主要 通过 使 用 GridView 控件 ,实现 在 网 格 中 显示 多 幅 图 像 。 当 实例 运行 之 后 ,在 
GridView 控件 中 将 显示 多 幅 图 像 , 单 击 其 中 任意 一 幅 图 像 , 则 在 弹出 的 Toast 中 显示 哪 幅 图 像 被 单 
击 了 ,效果 分 别 如 图 040. 1 的 左 图 和 右 图 所 示 。 
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图 040. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
private GridView myGridView = null; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myGridView = (GridView) findViewById(R. id.myGridView); 
myGridView. setAdapter(new ImageAdapter(MainActivity. this)); 
// 响 应 单 击 网 格 , 即 指明 单 击 了 哪 一 个 网 格 
myGridView. setOnItemClickListener(new AdapterView. OnItemClickListener() { 
public void onItemClick(AdapterView <?> arg0, View argl, int arg2, long arg3){ 
arg2 = arg2 + 1; 
Toast. makeText (MainActivity. this, 
"你 单 击 了 第 ”+ arg2 +" 幅 图 像 ", Toast.LENGTH_SHORT). show( ); 
} })7 } 
public class ImageAdapter extends BaseAdapter { 
private Context myContext; 
private int[ ] ImageBox = {R.drawable. sport1，R. drawable. moviel, 
R. drawable. bookl, R. drawable. moviel,R. drawable. moviel, 
R. drawable. bookl1, R. drawable. sportl1, R. drawable. moviel, 
R. drawable. sport1，R. drawable. moviel, R.drawable. moviel, 
R. drawable. bookl, R. drawable. moviel, R. drawable. sport1l, 
R. drawable. book1, R.drawable. moviel,R. drawable. sport1，R. drawable. bookl } ; 
public ImageAdapter(Context context) { this.myContext = context; } 
public int getCount() { return ImageBox. length; } 
public Object getItem( int arg0) { return null; } 
public long getItemId(int position) { return 0; } 
// 设 置 网 格 内 容 
public View getView( int position, View convertView, ViewGroup parent) { 
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ImageView myImageView; 

if (convertView == null) { 
myImageView = new ImageView(myContext); 
myImageView. setLayoutParams(new GridView. LayoutParams(260, 260)); 
myImageView. setScaleTypel( ImageView. ScaleType. CENTER CROP); 


myImageView. setPadding(8, 0, 0, 0); 
} else { myImageView = (ImageView) convertView; } 
myImageView. setImageResource( ImageBox[ position] ); 
return myImageView; 
FT 


上 面 这 段 代 码 在 MyCode\ MySample257\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 , myGridView. setAdapter (new ImageAdapter 
(MainActivity. this)) 用 于 设置 GridView 控件 的 数据 适配器 ,也 可 以 说 是 数据 源 , 即 GridView 控件 
将 要 加 载 的 内 容 。 除 此 之 外 ,GridView 控件 通常 还 需要 设置 列 数 和 行 数 等 基本 属性 。 

主要 代码 如 下 : 


< GridView android:id= "(@ + id/myGridView" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:columnWidth = "80dp" 
android:gravity = "center" 
android: horizontalSpacing = "6dp" 
android:numColumns = "auto_fit" 
android:paddingTop = "6dp" 
android: stretchMode = "columnWidth" 
android:verticalSpacing = "6dp"/> 


上 面 这 段 代 码 在 MyCode\MySample257\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: columnWidth 王 "80dp" 用 于 设置 列 宽 ,也 可 以 通过 android: numColumns 属 
性 设置 具体 的 列 数 。 当 某 个 属性 有 替代 方案 时 ,GridView 控件 能 够 进行 自动 计算 。 此 实例 的 完整 项 
日 在 MyCode\MySample257 文件 夹 中 。 


041 使 用 ViewPager 实现 缩放 轮 播 多 幅 图 像 


此 实例 主要 通过 使 用 onPageScrolled(int position，float positionOffset，int positionOffsetPixels) 方 法 
的 positionOffset 参数 调整 图 像 的 内 边 距 ,实现 ViewPager 在 轮 播 多 幅 图 像 时 , 进 场 图 像 根 据 当 前 位 置 
放大 ,退场 图 像 根 据 当 前 位 置 缩 小 的 动画 效果 。 当 实例 运行 之 后 ,7 个 红色 的 圆 点 表示 总 共有 7 幅 图 
像 ,白色 的 圆 点 表示 当前 显示 的 图 像 ; 使 用 手指 癌 左 滑动 , 则 左边 的 退场 图 像 缩 小 至 消失 ,右边 的 进 场 
图 像 放 大 至 满 屏 ,效果 分 别 如 图 041. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
private ViewPager myViewPager; 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myViewPager = (ViewPager) findViewById(R. id.myViewPager); 
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// 加 载 7 个 圆 点 指示 符 
LinearLayout myIndicators = (LinearLayout) findViewById(R. id. myIndicators); 
for (int i = 0; i<7; i++){ 
ImageView myPoint = new ImageView(this); 
LinearLayout. LayoutParams myParams = new LinearLayout.LayoutParams( 
LinearLayout. LayoutParams. WRAP CONTENT, 
LinearLayout. LayoutParams. WRAP CONTENT); 
myParams. setMargins(10, 10, 10, 10); 
myPoint. setImageResource(R. drawable. mypointer inactive); 
myPoint. setLayoutParams(myParams); 
myIndicators. addView(myPoint); 
} 
// 第 一 个 圆 点 为 白 点 表示 当前 显示 的 是 第 一 幅 图 像 
myIndicators. getChildAt(0). setBackgroundResource( 
R. drawable. mypointer _ active); 
MyAdapter myAdapter = new MyAdapter(this, myIndicators); 
myViewPager. setAdapter(myAdapter); 
myViewPager. addOnPageChangeListener(myAdapter); 
二 


MySample 


/2\ 两 


To 沁 


图 041. 1 


上 F 面 这 段 代 码 在 MyCode\ MySample696\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myViewPager. setAdapter(myAdapter) 表 示 使 用 目 定 义 
适 配 问 MyAdapter 管理 myViewPager 的 所 有 图 像 , 自 定义 适 配 问 MyAdapter 的 主要 代码 如 下 : 


public class MyAdapter extends PagerAdapter implements OnPageChangeListener{ 
public int myPosition = 0; 
public int myWidthPadding; 
public int myHeightPadding; 
public List <View> myViews = new ArrayList <View>(); 
public Context myContext; 
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public LinearLayout myIndicator; 
public MyAdapter(Context context, LinearLayout indicator) { 
myContext = context; 
InitViews( ); 
myIndicator = indicator; 
// 进 场 和 退场 缩放 图 像 的 (内 边 距 ) 宽 度 和 高 度 
myWidthPadding = 100; 
myHeightPadding = 150; 
} 
public Integer[ ] myImages = {R.mipmap.myimagel, R.mipmap.myimage2, 
R. mipmap. myimage3, R. mipmap. myimage4, R.mipmap. myimage5, 
R. mipmap. myimage6, R.nmipmap. myimage7}; 
private void InitViews() { 
For tin Tw 0 4 T= dir 
ImageView myImage = new ImageView(myContext); 
Bitmap myBitmap = 
BitmapFactory. decodeResource(myContext. getResources(), myImages[i]); 
myImage. setImageBitmap(myBitmap); 
myImage. setScaleTypel( ImageView. ScaleType. FIT XY); 
myViews.add(myImage); 
Eo 
Override 
public int getCount( ) {return myViews. sizel();} 
(Override 
public boolean isViewFromObject(View view, Object object) { return view == object;} 
(Override 
public void onPageScrolled( int position, float positionOffset, 
int positionOffsetPixels) { 
for (int i = 0; i< myViews. size(); i++) { 
myIndicator.getChildAt(i). setBackgroundResource( 
R. drawable. mypointer inactive); 
} 
if (position < myViews. size()) { 
myPosition = position; 
myIndicator. getChildAt(position). setBackgroundResource( 
R. drawable. mypointer active); 
// 退 场 缩小 图 像 
int outHeightPadding = (int) (positionOffset * myHeightPadding); 
int outWidthPadding = (int) (positionOffset * myWidthPadding); 
myViews. get (position). setPadding(outWidthPadding, 
outHeightPadding, outWidthPadding, outHeightPadding); 
// 进 场 放大 图 像 
if (position < myViews. size() - 1) { 
int inWidthPadding = (int) ((1 - positionOffset) * myWidthPadding); 
int inHeightPadding = (int) ((1 - positionOffset) * myHeightPadding); 
myViews. get(position + 1).setPadding(inWidthPadding, inHeightPadding, 
inWidthPadding, inHeightPadding); 
下 了 
(Override 
public void onPageSelected( int position) { } 
(Override 
public void destroyItem(ViewGroup container, int position, Object object) { 
container. removeView(myViews. get (position) ) ; 
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@Override 

public void onPageScrollStateChanged( int state) { } 

(@Override 

public Object instantiateItem(ViewGroup container, int position) { 
View myView = myViews. get(position); 
container. addView(myView); 
return myView; 


} } 


上 面 这 上段 代码 在 MyCode\ MySample696\app\src\ main\java\com\bin\luo\mysample\ 
MyAdapter. java 文件 中 。 在 这 段 代 码 中 ,myIndicator. getChildAt(i). setBackgroundResource(R. 
drawable. mypointer_inactive) 用 于 设置 红色 的 圆 点 ( 非 当 前 图 像 ) 指 示 右 ,mypointer_ inactive 是 
XML 文件, 它 是 采用 标签 的 形式 直接 绘制 圆 形 实现 的 。mypointer_inactive 文件 的 主要 内 容 如 下 : 


<?Xxml version= "1.0" encoding = "utf 一 8"?> 

< shape xmlns:android = "http://schemas. android. com/apk/res/android" 
android: shape = "oval"> 
< size android:width ="15dp" android:height = "15dp" /> 
< solid android:color = "#44FF0000" /> 

</ shape > 


上 面 这 段 代 码 在 MyCode\MySample696\app\src\main\res\drawable\mypointer_ inactive. xml 
文件 中 。 在 MyAdapter. java 文件 中 , mylIndicator. getChildAt (position). setBackgroundResource 
(R. drawable. mypointer_active) 用 于 设置 白色 的 圆 点 (当前 图 像 ) 指 示 器 ,mypointer_active 是 一 个 
XML 文件 , 它 也 是 采用 标签 的 形式 直接 绘制 圆 形 实现 的 。mypointer_active 文件 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 

< Shape xmlns:android = "http://schemas. android. com/apk/res/android" 
android: shape = "oval"> 
< size android:width = "15dp" android:height ="15dp" /> 
< Solid android:color =" #FFF"/> 

</shape > 


上 面 这 段 代 码 在 MyCode\MySample696\app\src\main\res\drawable\mypointer active. xml 文 
件 中 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: 
design: 23. 3.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample696 文件 夹 中 。 


042 ”使 用 Handler 实现 自动 轮 播 ViewPager 


此 实例 主要 通过 使 用 Handler 的 postDelayed() 方 法 延迟 显示 图 像 , 实 现 自 动 轮 播 在 ViewPager 
中 的 多 幅 图 像 。 当 实例 运行 之 后 ,在 ViewPager 中 的 多 幅 图 像 将 周而复始 地 自动 轮 播 ,效果 分 别 如 图 
042. 1 的 左 图 和 右 图 所 示 。 
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MySample MySample 


图 042.1 


public class MainActivity extends Activity { 
ViewPager myViewPager:; 
ViewPagerAdapter myViewPagerAdapter; 
Handler myHandler = new Handler( ) ; 
int[ ] myImages; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myViewPager = (ViewPager) findViewById(R. id.myViewPager); 
myImages = new int[ ]{R.mipmap.myimagel,R.mipmap.myimage2,R.mnmipmap.myimage3, 
R. mipmap. myimage4, R. mipmap. myimage5, R. mipmap. myimage6}; 
myViewPagerAdapter = new ViewPagerAdapter(MainActivity.this, myImages); 
myViewPager. setAdapter (myViewPagerAdapter); 
myViewPager. setOffscreenPageLimit (myImages. length); // 设 置 缓存 数量 
myHandler. postDelayed(new Runnable( ){ 
(Override 
public void run(){ 
myViewPager. setCurrentItem( (myViewPager. getCurrentItem( ) + 1) % myViewPager. 


getChildCount( )); // 自 动 显 示 下 一 幅 图 像 ,并 做 越界 处 理 
myHandler. postDelayed(this,1500); 
} },1500); // 每 间隔 1500ms 执行 一 次 


} 
public class ViewPagerAdapter extends PagerAdapter { 


private Context myContext; 

private int[ ] myImages; 

public ViewPagerAdapter(Context context, int[ ] datas)t{ 
mylImages = datas; 
myContext = context; 
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} 

Override 

public int getCount( ) { return myImages. length; } 

Override 

public boolean isViewFromObject(View view, Object object){return view == object;} 

(Override 

public Object instantiateItem(ViewGroup container, int position) { 
ImageView myImage = createlImageView(myContext, position); 
container. addView(myImage); 
return myImage; 

} 

(QOverride 

public void destroyItem(ViewGroup container, int position, Object object) { 
container. removeView( (View)object); 


} 
private ImageView createImageView(Context mContext, int position) { 


ImageView myImage = new ImageView(mContext); 

ViewPager. LayoutParams layoutParams = new ViewPager. LayoutParams(); 
myImage. setLayoutParams( layoutParanms); 

myImage. setImageResource(myImages[ position |); 

myImage. setScaleTypel( ImageView. ScaleType. CENTER CROP); 


return myImage; 
} } 
} 


上 面 这 段 代 码 在 MyCode\ MySample786 \app\src\ main\java\com\bin\luo\ mysampleA\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myHandler. postDelayed(this,1500) 表 示 间 隅 1500ms 显 
示 下 一 幅 图 像 ,postDelayed() 方 法 的 语法 声明 如 下 : 


public final boolean postDelayed(Runnable r, long delayMillis) 


其 中 ,参数 Runnable r 表示 要 延迟 执行 的 具体 任务 。 参 数 long delayMillis 表示 要 延迟 的 具体 
时 长 。 

需要 说 明 的 是 ,在 Android Studio 中 ,使 用 此 实例 的 相关 控件 需要 在 gradle 中 引入 compile 'com. 
android. support: design: 23. 3.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample786 文件 夹 中 。 


043 ”使 用 ViewPager 实现 华 果 风格 的 cover flow 


此 实例 主要 通过 在 ViewPager 控件 的 自 定 义 适 配器 中 为 图 像 添 加 倒影 ,并 在 该 控件 的 
setPageTransformer() 方 法 中 将 图 像 围 绕 y 轴 旋 转 指 定 的 角度 ,使 图 像 在 左右 滑动 时 实现 苹果 风格 的 
cover flow。cover flow 是 苹果 首创 的 将 多 首 歌曲 的 封面 以 3D 界面 的 形式 显示 出 来 的 方式 。 当 实例 
运行 之 后 ,左右 滑动 图 像 , 则 可 出 现 如 图 043. 1 的 左 图 和 右 图 所 示 的 cover flow 效果 。 

主要 代码 如 下 : 


public class MainActivity extends Activityf{ 
(Override 
protected void onCreate(Bundle savedInstanceState){ 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
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ViewPager myViewPager = (ViewPager)findViewById(R. id. myViewPager); 
MyAdapter myAdapter = new MyAdapter(this); 

myViewPager. setAdapter(myAdapter); 

ViewGroup. LayoutParams myParams = myViewPager. getLayoutParams( ); 
myParams. width = getWindowManager(). getDefaultDisplay().getWidth()/2; 
myViewPager. setLayoutParams(myParams); 


myViewPager. setOnPageChangeListener (myAdapter); // 设 置 滑动 监听 
myViewPager. setOffscreenPageLimit(3); // 设 置 缓存 图 像 数 量 
myViewPager. setPageTransformer(false, new ViewPager. PageTransformer( ){ 

(@ Override 


public void transformPage(View page,float position){ 
// 添 加 3D 偏 移 特效 (使 图 像 以 指定 的 角度 围绕 Y 轴 放置 ) 
page. setRotationY(position* - 30f); 
1 2 


MySample MySample 


图 043. 1 


上 面 这 段 代 码 在 MyCode\ MySample766\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myViewPager. setAdapter(myAdapter) 表 示 使 用 自 定义 
适配器 MyAdapter 管理 myViewPager 的 所 有 图 像 , 目 定义 适配器 MyAdapter 的 主要 代码 如 下 : 


public class MyAdapter extends 
PagerAdapter implements ViewPager. OnPageChangeListener { 
public List < ImageView> mylmageViewList; 
Integer[ ] myImageResources = {R.mipmap.myimagel, R.mipmap.myimage2, 
R. mipmap. myimage3, R. mipmap. myimage4, R.mipmap.myimage5, 
R. mipmap. myimage6, R.mipmap.myimage7}; 
public MyAdapter(Context context) { InitImageViewList(context); } 
(Override 
public int getCount() { return myImageViewList. size(); } 
(Override 
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public Object instantiateItem(ViewGroup container, int position) { 
int myPosition = position % myImageViewList. size( ); 
ImageView myImageView = mylmageViewList.get(myPosition); 
container.addView(myImageView); 
return myImageView; 
} 
(Override 
public void destroyItem(ViewGroup container, int position, Object object) { 
container. removeView( (View) object); 
} 
(Override 
public boolean isViewFromObject(View view, Object object){ return view == object; } 
public void InitImageViewList(Context context) { 
myImageViewList = new ArrayList < ImageView >(); 
for (int i = 0; i < myImageResources. length; i++) { 
ImageView myImageView = new ImageView(context); 
myImageView. setScaleType( ImageView. ScaleType. FIT XY); 
Bitmap myBitmap = BitmapFactory. decodeResource( 
context. getResources(), myImageResources[1i]); 
Bitmap myRef lectedBitmap = GenerateReflectedBitmap(myBitmap); 
myImageView. setImageBitmap(myRef lectedBitmap); 
myImageViewList.add(myImageView); // 添 加 图 像 及 其 倒影 
} } 
(Override 
public void onPageScrolled( int position, 
float positionOffset, int positionOffsetPixels) { 
} 
(Override 
public void onPageSelected( int position) { 
} 
(Override 
public void onPageScrollStateChanged( int state) { 
} 
public Bitmap GenerateReflectedBitmap(Bitmap bitmap) { // 生 成 倒影 图 像 
int myWidth = bitmap.getWidth( ); 
int myHeight = bitmap. getHeight( ); 
Matrix myMatrix = new Matrix(); 
myMatrix. preScale(1, - 1); // 创 建 缩 放 和 矩阵 
Bitmap myRef lectImagePart = Bitmap. createBitmap(bitmap, 0, 
myHeight / 2,myWidth, myHeight / 2, myMatrix, false); 
Bitmap myRef lectedImage = Bitmap. createBitmap(myWidth, 
(myHeight + myHeight / 2), Bitmap.Config.ARGB 8888); 
Canvas myCanvas = new Canvas(myReflectedImage); // 初 始 化 空白 画布 
myCanvas. drawBitmap(bitmap，0，0，nul1); // 绘 制 原 图 
myCanvas. drawBitmap(myReflectImagePart, 0, myHeight, null); // 绘 制 倒影 部 分 
Paint myShaderPaint = new Paint(); 
LinearGradient myShader = new LinearGradient(0, bitmap.getHeight(), 0, 
myRef lectedImage. getHeight(), Ox70ffffff, OxO0ffffff, Shader. TileMode. CLAMP); 
myShaderPaint. setShader(myShader); // 添 加 渐变 透明 效果 
myShaderPaint. setXfermode( new PorterDuffXfermode(PorterDuff. Mode.DST IN) ); 
myCanvas. drawRect(0, myHeight, myWidth, 
myRef lectedImage. getHeight(), myShaderPaint); 
return myRef lectedImage; 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample766 \app\src\ main\java\com\bin\luo\ mysample\ 
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MyAdapter. java 文件 中 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. 
android. support: design: 23. 3.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample766 文件 夹 中 ， 


044 使 用 RecyclerView 创建 水 平 瀑布 流 图 像 


此 实例 主要 通过 使 用 随机 数 设 置 在 列表 项 中 的 ImageView 控件 的 宽度 , 从 而 实现 使 用 
RecyclerView 显示 水 平 瀑布 流 样式 的 网 格 。 当 实例 运行 之 后 , 单 击 “显示 水 平 瀑布 流 网 格 对 话 框 ” 按 
钮 , 则 将 从 屏幕 底部 滑 出 一 个 瀑布 流 样 式 的 网 格 对 话 框 ,如 图 044. 1 的 左 图 所 示 。 在 水 平方 向 上 拖 动 
图 像 , 则 网 格 中 的 图 像 将 像 瀑 布 流 一 样 在 水 平方 向 上 滚动 ,如 图 044. 1 的 右 图 所 示 。 


主要 代码 如 下 : 


public class MainActivity extends AppCompatActivity { 
int myImages[ ] = {R. mipmap. myimagel, R. mipmap. myimage2, R. mipmap. myimage3, 
R. mipmap. myimage4, R. mipmap. myimage5, R. mipmap. myimage6, 
R. mipmap. myimage7, R. mipmap. myimage8, R. mipmap. myimage9 }; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity_main); } 
public void onClickmyBtnl (View v) { // 啊 应 单 击 "显示 水 平 瀑布 流 网 格 对 话 框 "按钮 
RecyclerView myRecyclerView = (RecyclerView)LayoutInflater. 
from(this). inflate(R. layout. dialogrecyclerview, null); 
List< Item> myList = new ArrayList <>(); 
for (int i = 0; i<100; i++) { 
Item myItem = new Item( ) ; 
int min = 0; 
int max = 8; 
Random random = new Random( ) ; 
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int num = random. nextInt(max) % (max— min+1) + min; 

myItem. setId(myImages[ num| ) ; 

myList. add(myItem) ; 
} 
ItemAdapter myAdapter = new ItemAdapter(myList); // 加 载 数据 
myRecyclerView. setAdapter(myAdapter); 
myRecyclerView. setItemAnimator(new DefaultItemAnimator( ) ) ; 
myRecyclerView. setLayoutManager( new StaggeredGridLayoutManager( 


5, StaggeredGridLayoutManager. HORIZONTAL) ) ; // 创 建 5 行 网 格 布局 
final BottomSheetDialog myBottomSheetDialog = new BottomSheetDialog(this); 
myBottomSheetDialog. setContentView(myRecyclerView); // 加 载 RecyclerView 
myBottomSheetDialog. show( ) ; // 显 示 底 部 滑 出 的 对 话 框 


} 
private class ItemHolder extends RecyclerView. ViewHolder { 


private ImageView myImageView; 
public ItemHolder(View itemView) { 
Super( itemView); 
int min = 50; 
int max = 350; 
Random random = new Random( ); 
int num = random.nextInt(max) % (max ~ min) + min; 
// 使 用 随机 数 设 置 myiten 布局 文件 的 InageView 控件 的 宽度 
myImageView = (ImageView) itemView.findViewById(R. id.myImageView); 
ViewGroup. LayoutParams myParams = myImageView. getLayoutParams( ); 
myParams. width = num; 
myImageView. SetLayoutParams(myParams ) ; 
} } 
public class Itenm { 
private int myID; 
public int getId() { return myID; } 
public void setId(int id) { this.myID = id; } 
} 
private class ItemAdapter extends RecyclerView. Adapter < ItemHolder > { 
private List < Item> myItems ; 
public ItemAdapter(List < Item> items) { myItems = items; } 
Override 
public ItemHolder onCreateViewHolder(ViewGroup viewGroup, int i) { 
LayoutInflater myInflater = LayoutInflater. from(MainRctivity.this); 
View myView = myInflater. inflate(R. layout. myitem, null); 
return new ItemHolder (myView); 
} 
(QOverride 
public void onBindViewHolder(final ItemHolder myHolder, int i) { 
final Item item = myItems. get(i); 
myHolder. myImageView. setImageResource( item. getId( )); // 设 置 图 像 资源 
} 
(Override 
public int getItemCount( ) { return myItems. size();} 


I 


上 面 这 段 代 码 在 MyCode\MySample492\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 ,myRecyclerView. setLayoutManager (new StaggeredGridLayoutManager(5， 
StaggeredGridLayoutManager. HORIZONTAL )) 表 示 在 水 平方 回 上 创建 5 行 网 格 。myRecyclerView = 
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(RecyclerView) LayoutInflater. from (this). inflate(R. layout. dialogrecyclerview ，null) 表示 根据 
dialogrecyclerview. xml 布局 创建 myRecyclerView。 关 于 dialogrecyclerview 布局 的 详细 内 容 请 参考 
源 代 码 中 的 MyCode\ MySample492\app\src\main\res\layout\dialogrecyclerview. xml 文件 。 

在 MainActivity. java 文件 中 ,myView 二 mylnflater. inflate(R. layout. myitem ,null) 表示 根据 
myitem 布局 设置 列表 项 内 容 。 关 于 myitem 布局 的 详细 内 容 请 参考 源 代 码 中 的 MyCodeA\ 
MySample492\app\src\main\res\layout\myitem. xml 文件 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需 
要 在 gradle 中 引入 compile 'com. android. support: design: 25. 0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 
MyCode\MySample492 文件 夹 中 。 


045 ”以 网 格 或 列表 显示 RecyclerView 列表 项 


此 实例 主要 通过 在 RecyclerView 的 setLayoutManager() 方 法 中 使 用 不 同 的 参数 ,从 而 实现 
RecyclerView 的 列表 项 以 网 格 样式 显示 ,或 者 以 列表 样式 显示 。 当 实例 运行 之 后 , 单 击 “ 网 格 显 示 ” 按 
钮 , 则 RecyclerView 的 列表 项 将 以 网 格 显示 ,如 图 045. 1 的 左 图 所 示 ; 单 击 “ 列 表 显 示 ” 按 钮 , 则 
RecyclerView 的 列表 项 将 以 列表 显示 ,如 图 045. 1 的 右 图 所 示 。 


MySample 


列表 显示 


图 045. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

public RecyclerView myRecyclerView; 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myRecyclerView = (RecyclerView) findViewById(R. id.myRecyclerView); 
myRecyclerView. setLayoutManager(new GridLayoutManager(this, 2)); 
myRecyclerView. setAdapter(new MyAdapter(this, true) ); 
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public void onClickmyBtnl(View v) { // 响 应 单 击 "网 格 显示 "按钮 
myRecyclerView. setLayoutManager(new GridLayoutManager(this, 2)); 
myRecyclerView. setAdapter(new MyAdapter(this, true) ); 
} 
public void onClickmyBtn2(View v) { // 响 应 单 击 "列表 显示 "按钮 
myRecyclerView. setLayoutManager(new LinearLayoutManager(this) ); 
myRecyclerView. setAdapter(new MyAdapter(this, false)); 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample653\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myRecyclerView. setAdapter (new MyAdapter (this， 
true) ) 表 示 使 用 自 定义 数据 适 配 右 MyAdapter 设置 RecyclerView 的 数据 源 , 自 定 义 数 据 适 配 融 
MyAdapter 的 主要 代码 如 下 : 


public class MyAdapter extends RecyclerView. Adapter < MyAdapter. MyViewHolder > { 
public boolean bGrid = false; 
public Context myContext; 
public String[ ] myFolders = {" 手 机 备份 "," 我 的 下 载 "," 我 的 文档 ", "我 的 照片 "}; 
public Integer[ ] myIcons = {R.mipmap.myicon phone, R.mipmap.myicon download, 
R. mipmap. myicon folder, R.mipmap.myicon camera}; 
public MyAdapter(Context context, boolean isGrid) { 
myContext = context; 
bGrid = isGrid; 
} 
(Override 
public MyViewHolder onCreateViewHolder(ViewGroup parent，int viewType) { 
MyViewHolder holder = null; 
if (bGrid) { // 加 载 网 格 时 的 Iten 布局 
holder = new MyViewHolder(LayoutInflater.from(myContext). 
inflate(R. layout. myrecyclerview griditem, parent, false)); 
} else { // 加 载 列 表 时 的 Iten 布局 
holder = new MyViewHolder(LayoutInflater.from(myContext). 
inflate(R. layout. myrecyclerview linearitem, parent, false)); 
} 
return holder; 
} 
(Override 
// 加 载 每 个 Item 
public void onBindViewHolder(MyViewHolder holder, int position) { 
holder. myTextView. setText(myFolders[ position |); 
holder. myImageView. setImageResource(myIcons[position]) ; 
(Override 
public int getItemCount( ) { return myFolders. length; } 
class MYViewHolder extends RecyclerView. ViewHolder { 
TextView myTextView; 
ImageView myImageView; 
public MYViewHolder(View view) { 
super (view); 
myImageView = (ImageView) view.findViewById(R. id.myImageView); 
myTextView = (TextView) view.findViewById(R. id.myTextView); 
}}] 
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上 F 面 这 段 代 码 在 MyCode\ MySample653\app\src\ main\java\com\bin\luo\mysample\ 
MyAdapter. java 文件 中 。 在 这 段 代 码 中 , holder = new MyViewHolder (LayoutInflater。from 


(myContext). inflate( R. layout. myrecyclerview_griditem,，parent,false)) 用 于 加 载 myrecyclerview_ 


griditem 布局 ,实现 Item 以 网 格 样式 显示 。myrecyclerview_ griditem 布局 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<LinearLayout xmlns:android = "http://schemas.android. com/apk/res/android" 
android:layout width= "match parent" 
android:layout height = "wrap_content" 
android:layout margin = "10dp" 
android:gravity = "center" 
android:orientation = "vertical"> 
< ImageView android:id="(@ + id/myImageView" 
android: layout width= "128dp" 
android:layout height = "128dp" 
android: scaleType = "fitXY" 
android:src = "(@mipmap/myicon folder"/> 
< TextView android:id= "(0 + id/myTextView" 
android: layout_width = "128dp" 
android:layout height = "wrap content" 
android:gravity = "center"/> 
</LinearLayout > 


上 面 这 段 代码 在 MyCode\MySample653\app\src\main\res\layout\myrecyclerview_ griditem. 
xml 文件 中 。 在 MyAdapter. java 文件 中 , holder = 二 new MyViewHolder (LayoutInflater. from 


(myContext). inflate( R. layout. myrecyclerview_linearitem，parent,false) ) 用 于 加 载 myrecyclerview 


_linearitem 布局 ,从 而 实现 Item 以 列表 样式 显示 。myrecyclerview_linearitem 布局 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 
<LinearLayout xmlns:android = "http://schemas.android. com/apk/res/android" 
android:layout width= "match parent" 
android:layout height = "wrap_content” 
android:layout margin = "10dp" 
android:orientation = "horizontal"> 
< ImageView android:id= "(0@ + id/myImageView" 
android: layout_width = "wrap_content" 
android: layout height = "32dp" 
android: scaleType = "fitCenter" 
android: src = "(@mipmap/myicon folder"/> 
< TextView android:id="(@ + id/myTextView" 
android:layout width= "match parent" 
android:layout height = "32dp" 
android:textSize = "18dp" 
android:gravity = "center vertical"/> 
</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample653\app\src\main\res\layout\myrecyclerview_ linearitem. 
xml 文件 中 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. 
support: recyclerview 一 v7: 25. 2.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample653 文件 
夹 中 。 
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046 ”使 用 RecyclerView 仿 表情 包 插 人 输入 框 


此 实例 主要 通过 使 用 自 定 义 RecyclerView. Adapter ,并 实现 View. OnClickListener 接口 ,从 而 实 
现 把 在 RecyclerView 中 选择 的 表情 图 像 插 入 到 输入 框 的 效果 。 当 实例 运行 之 后 ,在 输入 框 中 输入 文 
字 “ 好 极 了 ”, 再 在 下 面 的 RecyclerView 中 任意 选择 一 个 表情 , 则 该 表情 图 像 将 会 显示 在 输入 框 中 , 效 
果 分 别 如 图 046. 1 的 左 图 和 右 图 所 示 。 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
final EditText myEdit = (EditText) findViewById(R. id.myEdit); 
RecyclerView myRecyclerView = (RecyclerView) findViewById(R. id.myRecyclerView); 
myRecyclerView. setLayoutManager(new StaggeredGridLayoutManager(7, 
StaggeredGridLayoutManager. VERTICAL) ) ; // 限 制 每 行 显示 7 个 表情 图 像 
MyAdapter myAdapter = new MyAdapter(this); 
myAdapter. setOnItemClickListener( 
new MyAdapter. OnRecyclerViewItemClickListener() { 
(@ Override 
public void onItemClick(View view Integer data){ 
Bitmap myBitmap = BitmapFactory. decodeResource(getResources(), 
data); // 将 表情 图 像 转换 为 Bitmap 对 象 

ImageSpan myImageSpan = new ImageSpan(MainActivity. this, myBitmap); 

SpannableString myString = new SpannableString(" face" ); 

myString. setSpan(myInmageSpan, 0, 4, 

Spannable. SPAN EXCLUSIVE EXCLUSIVE); // 向 输入 框 插入 表情 图 像 
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myEdit. append( myString); 
上 
myRecyclerView. setAdapter(myAdapter); 
} 


上 面 这 上段 代码 在 MyCode\ MySample638\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,MyAdapter 是 自 定 义 的 RecyclerView. Adapter, 它 主要 用 
于 为 RecyclerView 管理 表情 图 像 ,MyAdapter 的 主要 内 容 如 下 所 示 : 


public class MyAdapter extends RecyclerView. Adapter < MyAdapter. MyViewHolder > 
implements View. OnClickListener { 
public Context myContext; 
private OnRecyclerViewItemClickListener myItemClickListener = null; 
public Integer[ ] myEmoji = {R.mipmap.myemojil, R.mipmap.myemoji2, 
R. mipmap. myemoji3, R.mipmap. myemoji4, R.mipmap. myemo]ji5, R.mipmap. myemoji6, 
R. mipmap. myemoji7, R.mipmap. myemoji8, R.mipmap. myemoji9, 


R. mipmap. myemoji10, R.mipmap.myemojill, R.mipmap. myemojil2 }; // 表 情 图 像 
public MyAdapter(Context context) { myContext = context; } 
(Override 


public MyAdapter. MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType){ 
View myView = 
LayoutInflater. from(myContext). inflate(R. layout. myitem, parent, false); 
MyViewHolder myHolder = new MyViewHolder(myView); 
myView. setOnClickListener(this); 
return myHolder; 
} 
(Override 
public void onClick(View v) { // 获 取 选 择 的 表情 图 像 
if (myItemClickListener != null) { 
myItemClickListener. onItemClick(v, (Integer) v.getTag()); 
bl 
(Override 
public void onBindViewHolder(MyAdapter. MyViewHolder holder，int position) { 
holder. myImageView. setImageResource(myEmoji[position|); 
holder. itemView. setTag(myEmoji[ position]); 
} 
(Override 
public int getItemCount() { return myEmoji. length; } 
public void setOnItemClickListener(OnRecyclerViewItemClickListener listener) { 
this.myItemClickListener = listener; 
} 
public class MyViewHolder extends RecyclerView.ViewHolder { 
ImageView myImageView; 
public MyViewHolder (View view) { // 加 载 所 有 表情 图 像 
super (view); 
myImageView = (ImageView) view.findViewById(R. id.myImageView); 
}]} 
public interface OnRecyclerViewItemClickListener { 
void onItemClick(View view, Integer data); 


1 


上 面 这 段 代 码 在 MyCode\ MySample638\app\src\ main\java\com\bin\luo\mysample\ 
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MyAdapter. java 文件 中 。 在 这 段 代 码 中 ,myView 二 LayoutInflater. from (myContext). inflate(R. 
layout. myitem，parent,false) 用 于 加 载 RecyclerView 的 每 个 Item 的 布局 ,此 处 的 myitem 非常 简单 ， 
它 只 有 一 个 ImageView 控件 , 即 显 示 一 个 表情 图 像 ; 实际 上 ,如 果 将 Item 布局 设计 得 较为 复杂 ， 
RecyclerView 就 会 呈现 更 加 丰富 的 效果 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 
compile 'com. android. support: recyclerview 一 v7: 25. 3.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample638 文件 夹 中 ， 


047 ”使 用 CardView 显示 RecyclerView 列表 项 


此 实例 主要 实现 了 在 单 击 RecyclerView 的 任 一 列表 项 时 ,以 动画 的 形式 动态 在 CardView 中 显 
示 单 击 的 列表 项 。 当 实例 运行 之 后 , RecyclerView 的 所 有 列表 项 如 图 047. 1 的 左 图 所 示 ; 单 击 
RecyclerView 的 任 一 列表 项 , 则 将 以 动画 的 形式 弹出 CardView ,并 在 其 中 显示 此 列表 项 ,同时 背景 模 
糊 , 如 图 047. 1 的 右 图 所 示 。 


MySample 


@ 国 沪 析 Hadoop 中 的 数据 倾 人 


| 和 -UD 


外 


图 047.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

private FrameLayout myRootView; 

private CardView myCardView; 

private MyAdapter myAdapter; 

private RecyclerView myRecyclerView; 

(@ Override 

protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity_main); 
myCardView = (CardView) findViewById(R. id.MyCardView); 
myRootView = (FrameLayout) findViewById(R. id.activity main); 
myRecyclerView = (RecyclerView) findViewBylId(R. id.MyRecyclerView); 
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myAdapter = new MyAdapter(this); 
myRecyclerView. setAdapter(myAdapter); 
myRecyclerView. setLayoutManager(new LinearLayoutManager (this) ); 
myAdapter. setOnItemClickListener(new MyAdapter. OnItemClickListener() { 
(Override 
public void onItemClick(View view, int position){ // 单 击 列表 项 
final View myMainView = getWindow( ). getDecorView( ); 
myMainView. setDrawingCacheEnabled(true); 
myMainView. buildDrawingCache( ); 
Bitmap myOriginBitmap = myMainView.getDrawingCache( ); 
Rect myFrame = new Rect( ); 
getWindow( ) . getDecorView( ).getWindowVisibleDisplayFrame(myFrame); 
int myStatusBarHeight = myFrame. top; 
int myScreenWidth = getWindowManager().getDefaultDisplay().getWidth( ); 
int myScreenHeight = getWindowManager( ). getDefaultDisplay(). getHeight(); 
Bitmap myClipImage = Bitmap.createBitmap(myOriginBitmap,0, 
myStatusBarHeight, myScreenWidth, myScreenHeight - myStatusBarHeight); 
myRootView. setBackground(new BitmapDrawable(getResources( ), 


MyUtils. doBlur (myClipImage, 20))); // 使 用 模糊 的 截图 设置 背景 
myMainView. setDrawingCacheEnabled(false); 
myRecyclerView. setVisibility(GONE); // 隐 藏 nyRecyclerView 
showView(view, position); // 使 用 CardView 显示 选择 的 Item 


三， 
(Override 
public boolean onKeyDown( int keyCode, KeyEvent event) { 
if (keyCode == KeyEvent.KEYCODE BACK) { 


myRecyclerView. setVisibility(View. VISIBLE); // 显 示 myRecyclerView 
myRootView. setBackgroundColor (Color. WHITE); // 主 窗口 设置 白色 背景 
myCardView. setVisibility(GONE ) ; // 隐 藏 nyCardView 
myCardView. removeAllViews( ); 

return true; 


} 
return super. onKeyDown( keyCode, event ) ; 
} 
// 使 用 CardView 显示 对 应 的 Item 
private void showView(View view int position){ 
View myItemView = LayoutInflater.from(this).inflate(R. layout.myitem, null); 
TextView myTextView = (TextView)myItemView.findViewById(R. id.myTextView); 
myTextView. setText(myAdapter. getMYData( ) . get(position) ); 
// 设 置 myCardView 样式 ,位 置 通 过 margintop 来 计算 
FrameLayout. LayoutParams myParams = 
new FrameLayout. LayoutParams(view. getWidth() -~ 30, view.getHeight()); 
myParams.topMargin = (int)view.getY(); 
myParams. leftMargin = 15; 
myParams.rightMargin = 15; 
myCardView. setVisibility(View.VISIBLE); 
myCardView. setLayoutParams( myParams); 
// 在 CardView 上 加 载 myItemView, 并 设置 样式 为 iten 样式 
myCardView.addView(myItemView, view.getLayoutParams( ) ); 
startAnimate(myCardView); // 显 示 myCardView 动画 
} 
private void startAnimate(CardView cardView) { 
PropertyValuesHolder myOutScaleXHolder = 


} 
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PropertyValuesHolder. ofFloat(" scaleXx", 0.1f, 1.05f); 
PropertyValuesHolder myOutScaleYHolder = 

PropertyValuesHolder. ofFloat(" scaleY", 0.1f, 1.05f); 
ObjectAnimator myOutAnim = ObjectAnimator.ofPropertyValuesHolder( 

myCardView, myOutScaleXHolder, myOutScaleYHolder ); 
myOutAnim. setInterpolator(new AccelerateDecelerateInterpolator( ) ); 
myOutAnim. setDuration( 350); 
PropertyValuesHolder myInScaleXHolder = 

PropertyValuesHolder. ofFloat(" scalex", 1.05f, 1f); 
PropertyValuesHolder myInScaleYHolder = 

PropertyValuesHolder. ofFloat(" scaleY", 1.05f, 1f); 
ObjectAnimator myInAnim = ObjectAnimator. ofPropertyValuesHolder( 

myCardView, myInScaleXHolder,myInScaleYHolder ); 
myInAnim. setInterpolator(new AccelerateDecelerateInterpolator( ) ); 
myInAnim. setDuration(100); 
RnimatorSet myAnimSet = new AnimatorSet( ); 
myAnimSet. playSequentially(myOutAnim, myInAninm); // 按 顺 序 执行 两 个 动画 
myAnimSet. start(); 
} 


上 面 这 段 代 码 在 MyCode\ MySample731 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myRecyclerView. setAdapter(myAdapter) 表 示 使 用 自 定 
义 适 配 需 MyAdapter 的 实例 myAdapter 管理 myRecyclerView 的 列表 项 , 自 定 义 适 配器 MyAdapter 
的 主要 代码 如 下 : 


public class MyAdapter extends RecyclerView. Adapter < MyAdapter. MyViewHolder > { 
public Context myContext; 
public interface OnItemClickListener {void onItemClick(View view, int position); } 
public String[ ] myItems = {【 技 术 公 开课 Python 数据 分 析 与 机 器 学 习 实 战 !"， 
"< 程序 员 >4 月 精彩 内 容 :分 布 式 数据 库 应 用 实践 "，" 浅 析 Hadoop 中 的 数据 倾斜 "， "百度 通 用 AI 大 突破 ,智能 体 


通 


过 交互 式 学 习 实现 举一反三 "，"LinkedIn 增长 揭秘 :262 亿美 元 的 增长 引擎 是 如 何 练 成 的 ?","【 技 术 直 播 ] 大 


数据 技术 企业 应 用 实战 !1"," 机 器 学 习 里 的 贝 叶 斯 基本 理论 模型 和 算法 ","Android 内 存 优化 实践 与 总 结 "}; 
public OnItemClickListener myListener; 
public List< String> myData = new ArrayList< String>(); 
public MyAdapter(Context context) { 


} 


myContext = context; 
for (int i = 0; i< myItems. length; i++) { myData. add(myItems[i]); } 


public List < String> getMyData( ) { return myData; } 
public void setOnItemClickListener(OnItemClickListener listener) { 


} 


this.myListener = listener; 


(Override 
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { 


} 


MyViewHolder myHolder = new MyViewHolder(LayoutInflater. from( 
myContext). inflate(R. layout. myitem, parent, false) ); 
return myHolder; 


(@Override 
public void onBindViewHolder(MyViewHolder holder, final int position) { 


} 


holder. myTextView. setText(myData. get(position) ); 


(Override 
public int getItemCount( ) { return myData. size(); } 
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class MyViewHolder extends RecyclerView.ViewHolder { 
TextView myTextView; 
public MyViewHolder(View view) { 
super (view); 
myTextView = (TextView) view.findViewById(R. id.myTextView); 
View. setOnClickListener(new View. OnClickListener() { 
WOverride 
public void onClick(View view) { 
myListener. onItemClick(view, getAdapterPosition( ) ); 
F731 


上 面 这 段 代 码 在 MyCode\ MySample731 \app\src\ main\java\com\bin\luo\mysample\ 
MyAdapter. java 文件 中 。 在 这 段 代 码 中 ,myHolder = 二 new MyViewHolder(LayoutlInflater. from 
(myContext). inflate(R. layout. myitem，parent,false) ) 用 于 加 载 myitem 布局 以 创建 MyAdapter 的 
每 个 Item 容器 , 即 每 个 列表 项 的 布局 。 关 于 myitem 布局 的 详细 内 容 请 参考 源 代 码 中 的 MyCode\ 
MySample731\app\src\main\res\layout\myitem. xml 文件 。 在 MainActivity. java 文件 中 ,MyUtils. 
doBlur(myClipImage,20) 主 要 用 于 模糊 主 窗口 截图 , MyUtils 是 自 定 义 类 ,代码 较 多 ,详细 内 容 请 查 
看 源 代 码 中 的 MyCode\MySample731\ app\src\main\java\com\bin\luo\mysample\MyUtils. java 文 
件 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: 
recyclerview 一 v7: 25. 3. 1' 和 compile 'com. android. support: cardview 一 v7: 25. 3. 1' 依 赖 项 。 此 实例 
的 完整 项 目 在 MyCode\MySample731 文件 夹 中 。 


048 在 ListView 中 创建 图 文 结合 列表 项 


此 实例 主要 通过 在 布局 文件 中 为 每 个 列表 项 添加 图 像 和 文本 等 控件 ,实现 在 ListView 中 显示 图 
文 混合 的 列表 项 。 当 实例 运行 之 后 ,ListView 的 每 个 列表 项 将 混合 显示 图 像 和 文本 ,每 个 列表 项 之 间 
用 细 线 分 隔 , 单 击 任意 列表 项 , 则 在 弹出 的 Toast 中 显示 该 列表 项 的 内 容 , 效 果 分 别 如 图 048. 1 的 左 
图 和 右 图 所 示 。 


MySample MySample 


神秘 宇宙 与 微观 世界 珍贵 图 集 神秘 宇宙 与 微观 世界 珍贵 图 集 
北京 联合 出 版 公司 北京 联合 出 版 公司 


写 给 所 有 人 的 极 简 统计 学 写 给 所 有 人 的 极 简 统计 学 
北京 时 代 华 文书 局 北京 时 代 华 文书 局 


很 杂 很 杂 的 杂 学 知识 很 杂 很 杂 的 杂 学 知识 


中 国画 报 出 版 社 中 国画 报 出 版 社 


英国 皇家 园艺 学 会 植物 学 图 集 英国 旦 家 园艺 学 会 植物 学 图 集 
重庆 大 学 出 版 社 


您 选择 的 图 书 是 : 很 杂 很 杂 的 杂 学 知 
识 ， 出 版 公司 是 : 中 国画 报 出 版 社 


图 048. 1 
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主要 代码 如 下 : 


< RelativeLayout android:layout width= "fil1 _ parent” 
android:layout height = "?android:attr/listPreferredItemHeight"> 
< ImageView android: id = "(0 + id/myImage" 
android: layout width= "wrap_ content" 
android:layout_ height = "fill parent" 
android: layout alignParentBottom = "true" 
android:layout alignParentTop = "true" 
android:adjustViewBounds = "true" 
android:padding = "5dip"/> 
<TextView android: id = "(0 + id/myName" 
android:layout width = "wrap_content" 
android: layout_ height = "wrap_content" 
android:layout above = "(@ + id/myPress" 
android:layout alignParentRight = "true" 
android: layout alignParentTop = "true" 
android:layout alignWithParentIfMissing = "true" 
android: layout toRightOf = "(@ + id/myImage" 
android:gravity = "center vertical" 
android:textSize = "22dip"/> 
<TextView android:id = "(@ + id/myPress" 
android:layout width= "fill parent" 
android:layout height = "wrap_content" 
android: layout alignParentBottom = "true" 
android:layout alignParentRight = "true" 
android: layout toRightOf = "(@ + id/myImage" 
android:ellipsize = "marquee" 
android:singleLine = "true" 
android: textSize = "24dip"/> 
</RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample068\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代码 中 ,所 有 控件 组 合 在 一 起 仅 表示 ListView 的 一 个 列表 项 ,ListView 的 所 有 列表 项 则 通过 
Adapter 来 映射 实现 ,主要 代码 如 下 : 


public class MainActivity extends ListActivity { 
private String[ ] myNameList = {" 神 秘 宇 宙 与 微观 世界 珍贵 图 集 "," 写 给 所 有 人 的 极 简 统计 学 ", "很 杂 很 杂 的 
杂 学 知识 "," 英 国 皇家 园艺 学 会 植物 学 图 集 "}; 
private String[ ] myPressList = {" 北 京 联合 出 版 公司 ", "北京 时 代 华 文书 局 "," 中 国画 报 出 版 社 ", "重庆 大 
学 出 版 社 "}; 
ArrayList <Map < String, Object >> myData = new ArrayList <Map< String, Object >>(); 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
int myLength = myNameList. length; 
for (int i = 0; i< myLength; i++) { // 向 每 个 列表 项 添加 内 容 
Map < String, Object > myItem = new HashMap < String, Object >(); 
myItem. put(" image", R.mipmap. mybook); 
myItem. put("name", myNameList[i]); 
myItem. put("press", myPressList[i]); 
myData. add( myItem); 
} 
SimpleAdapter myAdapter = new SimpleAdapter(this, myData, 


CA > Anaroia 引 应 有 300 。 实战 篇 


R. layout. activity mainr new String[ ]{" image", "name", "press" }, 
new int[ ]{R. id.myImage, R. id.myName, R. id.myPress} ); 
setListAdapter(myAdapter); // 实 现 列 表 项 数据 映射 
ListView myList = getListView( ); 
myList. setChoiceMode(ListView.CHOICE MODE SINGLE); // 设 置 列表 项 单 选 模式 
// 为 列表 项 添加 单 击 事件 响应 方法 
myList. setOnItemClickListener(new OnItemClickListener() { 
(@ Override 
public void onItemClick(AdapterView <?> adapterView, 
View view, int position, long id) { 
Toast. makeText(MainActivity.this, "您 选择 的 图 书 是 :”+ 
myNameList[position] + ", 出 版 公司 是 :"” + 
myPressList[position], Toast.LENGTH LONG). show( ); 
Ps 


super. onCreate(savedInstanceState) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySample068\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,setListAdapter(myAdapter) 用 于 把 数据 映射 到 界面 里 边 ， 
ListView 显示 的 数据 必须 借助 Adapter 来 映射 。getListView() 方 法 用 于 获取 当前 环境 的 ListView 
对 象 。setChoiceMode (ListView. CHOICE _ MODE _ SINGLE) 表示 ListView 支持 单行 选择 。 
setOnItemClickListener() 主要 是 为 单行 选择 添加 监听 事件 啊 应 方法 。 此 实例 的 完整 项 目 在 MyCode 
\MySample068 文件 夹 中 ， 


049 使 用 ListPopupWindow 实现 下 拉 选 择 


此 实例 主要 通过 在 ListPopupWindow 中 添加 列表 选项 ,实现 为 EditText 控件 添加 类 似 于 上 自动 完 
成 的 选择 功能 。 当 实例 运行 之 后 , 单 击 “报考 专业 :” 输 入 框 , 则 将 弹出 一 个 列表 选项 窗口 , 单 击 其 中 的 
任意 一 个 选项 ,如 “计算 机 科学 与 技术 ”, 则 该 选项 将 自动 填充 到 “报考 专业 :” 输 入 框 中 ,如 图 049. 1 的 
左 图 和 右 图 所 示 。 


MySample MySample 


报考 专业 : 计算 机 科学 与 技 村 报考 专业 : 数学 与 应 用 数学 


计算 机 科学 与 技术 计算 机 科学 与 技术 

机 械 设计 制造 及 其 自动 化 机 械 设计 制造 及 其 自动 化 
电气 工程 及 其 自动 化 电气 工程 及 其 自动 化 
电子 信息 工程 电子 信息 工程 

数学 与 应 用 数学 数学 与 应 用 数学 


国际 经 济 与 贸易 国际 经 济 与 贸易 


图 049.1 


第 2 章 ”常用 控件 人 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

Private EditText myEditText; 

private ListPopupWindow myPopupWindow; 

private List < String> myArray = new ArrayList(); 

Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
myArray.add(" 计 算 机 科学 与 技术 "); 
myArray.add(" 机 械 设计 制造 及 其 自动 化 "); 
myArray.add(" 电 气 工 程 及 其 自动 化 "); 
myArray.add(" 电 子 信息 工程 "); 
myArray. add(" 数 学 与 应 用 数学 "); 
myArray.add(" 国 际 经 济 与 贸易 "); 
myEditText = (EditText) findViewById(R. id.myEditText); 
myPopupWindow = new ListPopupWindow(this); 
myPopupWindow. setAdapter(new ArrayAdapter(this, 

android. R. layout. simple list item 1, myArray)); 

myPopupWindow. setWidth(ViewGroup. LayoutParams. WRAP CONTENT); 
myPopupWindow. setHeight (ViewGroup. LayoutParams. WRAP_CONTENT); 
// 设 置 ListPopupWindow 的 锚 点 , 即 关 联 PopupWindow 的 显示 位 置 和 这 个 锚 点 
myPopupWindow. setRnchorView(myEditText); 


myPopupWindow. setModal( true); // 设 置 模式 
myPopupWindow. setOnItemClickListener(new AdapterView. OnItemClickListener() { 
(@ Override 


public void onItemClick(AdapterView Parent， 

View view, int position, long id) { 
myEditText. setText (myArray. get(position). toString( ) ) ; 
myPopupWindow. dismiss( ); 

} }); 
myEditText. setOnClickListener(new View. OnClickListener() { 
(@ Override 
public void onClick(View v) { myPopupWindow. show(); } 
}); 
3 


上 面 这 段 代 码 在 MyCode\ MySample431 \app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPopupWindow. setAnchorView (myEditText) 用 于 将 
ListPopupWindow 锚 定 在 EditText 控件 上 ,即将 两 者 关联 起 来 。 此 实例 的 完整 项 目 在 MyCode\\ 
MySample431 文件 夹 中 。 


050 ”使 用 Elevation 创建 阴影 扩散 的 控件 


此 实例 主要 通过 设置 elevation 属性 或 使 用 setElevation() 方 法 ,实现 给 控件 设置 扩散 阴影 。 当 实 
例 运行 之 后 , 单 击 “ 设 置 扩 散 阴 影 * 按 钮 , 则 图 像 的 四 周 将 有 一 层 灰 色 的 扩散 阴影 ,如 图 050. 1 的 左 图 
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所 示 。 单 击 “ 移 除 扩散 阴影 ?按钮 , 则 将 移 除 图 像 四 周 的 扩散 阴影 ,如 图 050. 1 的 右 图 所 示 。 


sl™ 中 dv 


MySample 


设置 扩散 阴影 移 除 扩散 阴影 设置 扩散 阴影 移 除 扩散 阴影 


图 050. 1 
主要 代码 如 下 : 


// 响 应 单 击 "设置 扩散 阴影 "按钮 
public void onClickmyBtnl (View v) { 
myImageView. setElevation( 35); 


} 

// 响 应 单 击 " 移 除 扩散 阴影 "按钮 

public void onClickmyBtn2(View v) { 
myImageView. setElevation(0); 


} 


上 面 这 段 代 码 在 MyCode\ MySample293\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImageView. setElevation(35) 用 于 设置 图 像 四 周 的 阴 
影 扩 散 尺 寸 为 35 像素 ,在 XML 文件 中 ,直接 设置 控件 的 android: elevation 属性 即 可 。 需 要 注意 的 
是 ,在 设置 android: elevation 属性 时 ,应 该 同时 设置 android: background 属性 。 此 实例 的 完整 项 目 
在 MyCode\MySample293 文件 夹 中 ， 


051 在 单 击 CheckBox 时 显示 波纹 扩散 效果 


此 实例 主要 通过 设置 CheckBox 控件 的 android: background 属性 为 "? android: attr/ 
selectableItemBackground" ,实现 在 单 击 CheckBox 控件 时 产生 有 界 和 矩形 的 波纹 扩散 动画 效果 。 当 实 
例 运 行 之 后 , 单 击 前 面 3 个 CheckBox 控件 ,将 会 产生 默认 的 圆 形 无 界 的 中 心 回 四 周 扩 散 的 波纹 动画 
效果 ,如 图 051. 1 的 左 图 所 示 ; 单 击 第 4 个 CheckBox 控件 , 则 产生 定制 的 矩形 有 界 的 中 心身 四 周 扩 
散 的 波纹 动画 效果 ,如 图 051. 1 的 右 图 所 示 。 


口 
口 
OQ 


主要 代码 如 下 : 
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MySample 
请 评选 服务 最 满意 的 公司 : 
国家 电网 公司 口 国家 电网 公司 
中 国 石 油 天 然 气 集团 公司 口 中 国 石油 天 然 气 集团 公司 
中 国 石油 化 工 集团 公司 口 中 国 石油 化 工 集团 公司 
中 国 工商 银行 股份 有 限 公 司 中 国 工商 银行 股份 有 限 公 司 


图 051. 1 


<LinearLayout android:layout width = "match parent” 
android:layout height = "match parent" 
android:orientation = "vertical" 
android:padding = "10px"> 


< TextView 


< CheckBox 


< CheckBox 


< CheckBox 


< CheckBox 


android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 
android 


android 
android 
android 
android 
android 
android 


:layout width = "wrap_content" 
:layout height = "wrap_content" 
:layout marginBottom = "10dp” 
:text = "请 评选 服务 最 满意 的 公司 :" 
:textSize = "22dp" /> 

: id = "(@ + id/checkboxl1" 
:layout_width = "wrap_content" 
:layout height = "wrap_content" 
:textSize = "20dp" 

:text = "国家 电网 公司 "/> 

:id="(@ + id/checkbox2" 

:layout width= "wrap_content" 
:layout height = "wrap_content" 
:textSize = "20dp" 

:text = "中 国 石油 天 然 气 集团 公司 "/> 
:id = "(@ + id/checkbox3" 
:layout_width = "wrap_content" 
:layout height = "wrap_content" 
:textSize = "20dp" 


:background = 


"?android:attr/selectableItemBackgroundBorderless" 
:text = "中 国 石油 化 工 集团 公司 "/> 
:id = "(@ + id/checkbox4" 
:1ayout_width = "wrap_content" 
:layout height = "wrap_content” 
:textSize = "20dp" 


:background = " ?android:attr/selectableItemBackground" 


常用 控件 
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android:text = "中 国 工商 银行 股份 有 限 公司 "/> 
</LinearLayout > 


上 面 这 段 代码 在 MyCode\MySample570\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: background 王 "? android: attr/selectableltem BackgroundBorderless" 表 示 
在 单 击 checkbox3 控件 时 ,将 会 产生 从 圆 形 无 界 的 中 心 回 四 周 扩散 的 波纹 动画 效果 ,这 也 是 默认 的 单 
击 控件 效果 ,不 设置 此 属性 也 会 产生 这 种 效果 。android: background 一 "? android: attr/ 
selectableItemBackground" 用 于 实现 在 单 击 checkbox4 控件 时 产生 有 界 和 矩形 边框 限制 的 波纹 扩散 动 
男 效 果 。 此 实例 的 完整 项 目 在 MyCode\MySample570 文件 夹 中 。 


052 ”使 用 自 定义 形状 定制 Switch 开关 状态 
此 实例 主要 通过 以 自 定义 形状 设置 thumb 属性 和 track 属性 ,实现 定制 Switch 控件 的 开关 状态 。 
当 实 例 运 行 之 后 ,空心 圆 位 于 Switch 控件 的 右 端 ,整个 控件 顶部 居中 ,如 图 052. 1 的 左 图 所 示 ; 单 击 


Switch 控件 右 端 的 空心 加 , 右 端的 空心 圆 立即 滑动 到 Switch 控件 的 左 端 ,整个 控件 底部 居中 ,如 图 
052. 1 的 右 图 所 示 。 


MySample MySample 


顶部 居中 国 


图 052.1 
主要 代码 如 下 : 


< Switch android: id = "(@ + id/mySwitch" 
android:layout width= "wrap_content” 
android:layout height = "wrap content" 
android:checked = "true" 
android:text = "顶部 居中 " 
android:textSize = "20dp" 
android:thumb = " (Odrawable/mythumb" 
android:track = " (Wdrawable/mytrack" /> 
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上 面 这 段 代 码 在 MyCode\MySample054\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: thumb="(@drawable/mythumb" 表 示 Switch 控件 的 thumb( 滑 动 的 空心 圆 ) 
采用 mythumb 资源 进行 配置 。android: track 王 "(@ drawable/mytrack" 表 示 Switch 控件 的 track( 紫 
色 / 灰 色 的 轨道 ) 采 用 mytrack 资源 进行 配置 。mythumb 是 一 个 XML 文件 ,该 文件 的 主要 内 容 如 下 : 


<?Xxml version= "1.0" encoding = "utf 一 8"?> 

< Shape xmlns:android = "http://schemas.android. com/apk/res/android" 
android: shape = "oval"> 

< Size android:width ="30dp”android:height = "30dp" /> 

< solid android: color = "(Wandroid: color/white" /> 

</shape > 


上 面 这 段 代 码 在 MyCode\MySample054\app\src\main\res\drawable\mythumb. xml 文件 中 。 
在 这 段 代 码 中 ,android: shape 王 "oval" 指 定 thumb 的 形状 为 椭圆 ; 当 android: width 王 "30dp"， 
android: height 王 "30dp" 时 ,此 椭圆 即 是 一 个 圆 。android: color 二 "(@android: color /white" 表示 圆 
的 填充 颜色 是 白色 ,因此 看 起 来 像 是 一 个 空心 圆 。 

mytrack 也 是 一 个 XML 文件, 而且 是 一 个 selector 文件 。 该 文件 的 主要 内 容 如 下 : 


<?Xxml version= "1.0" encoding = "utf 一 8"?> 
< selector xmlns:android = "http://schemas.android. com/apk/res/android"> 
< item android: state checked = "true" android:drawable = "(Odrawable/mytrackon" /> 
< item android: state checked = "false" android:drawable = "(Odrawable/mytrackoff" /> 


</selector > 


上 面 这 段 代 码 在 MyCode\MySample054\app\src\main\res\drawable\mytrack. xml 文件 中 。 在 
这 段 代 码 中 ,< item android: state_checked 王 "true" android: drawable 王 "(@ drawable /mytrackon"/> 
表示 当 Switch 的 state_checked 属性 值 为 true 时 ,设置 mytrackon 资源 。< item android: state 
checked= 二 "false" android: drawable = 二"(@drawable/ mytrackoff" /表示 当 Switch 的 state_checked 
属性 值 为 false 时 ,设置 mytrackoff 资源 。mytrackoff 资源 和 mytrackon 资源 都 是 XML 文件 ， 
mytrackon 文件 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< shape xmlns:android= "http://schemas.android. com/apk/res/android" 
android: shape = "rectangle"> 
< solid android: color = "(Wandroid: color/holo purple"/> 
< Corners android: radius = "30dp"/> 
</shape> 


上 面 这 段 代 码 在 MyCode\MySample054\app\src\main\res\drawable\mytrackon. xml 文件 中 。 
在 这 段 代 码 中 ,android: shape 王 "rectangle" 表 示 Switch 在 打开 状态 时 的 track 是 一 个 rectangle。 
< solid android: color= 王 "人 (android: color/yholo_purple" /> 表示 该 rectangle 是 一 个 紫色 的 rectangle。 
< corners android: radius 王 "30dp" /> 表示 该 rectangle 的 4 个 直角 的 圆 角 半径 是 30dp。mytrackoff 文 
件 的 主要 内 容 如 下 : 
<?xml] version= "1.0" encoding = "utf 一 8"?> 


< shape xmlns:android = "http://schemas. android. com/apk/res/android" 
android: shape = "rectangle"> 
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< solid android: color = "(Wandroid: color/darker gray" /> 
< corners android: radius ="30dp" /> 
</shape> 


上 了 面 这 段 代 码 在 MyCode\MySample054\app\src\main\res\drawable\mytrackoff. xml 文件 中 。 
在 这 段 代 码 中 ,android: shape 王 "rectangle" 表 示 Switch 在 关闭 状态 时 的 track 是 一 个 rectangle。 
< solid android: color 一 "人 (android: color/darker_gray"/> 表 示 该 rectangle 是 一 个 灰色 的 rectangle。 
< corners android: radius 一 "30dp"/> 表 示 该 rectangle 的 4 个 直角 的 圆 角 半径 是 30dp 。 

Switch 的 CheckedChange 事件 的 啊 应 方法 则 通过 Java 代码 实现 ,主要 代码 如 下 : 


public class MainActivity extends Activity { 
private Switch mySwitch; 
private LinearLayout myLayout; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) 
setContentView(R. layout. activity main); 
mySwitch = (Switch) findViewById(R. id.mySwitch); 
myLayout = (LinearLayout) findViewById(R. id.myLayout); 
mySwitch. setOnCheckedChangeListener( 
new CompoundButton. OnCheckedChangeListener() { 
(Override 
public void onCheckedChanged( CompoundButton buttonView, boolean isChecked) { 
if (isChecked) { 
myLayout. setGravity(Gravity. CENTER HORIZONTAL); 
mySwitch. setText(" 顶 部 居中 "); 
} else { 
myLayout. setGravity(Gravity. CENTER HORIZONTAL | Gravity.BOTTOM); 
mySwitch. setText(" 底 部 居中 " ); 
F211 


上 上面 这 段 代 码 在 MyCode\ MySample054\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myLayout. setGravity(Gravity. CENTER_ HORIZONTAL ) 用 
于 设置 myLayout 布局 管理 器 顶部 大 中。myLayout. setGravity(Gravity. CENTER_HORIZONTAL 
| Gravity. BOTTOM) 用 于 设置 myLayout 布局 管理 器 底部 居中。 此 实例 的 完整 项 目 在 MyCode\ 
MySample054 文件 夹 中 ， 


053 ” 自 定 义 selector 以 渐变 前 景 切换 控件 


此 实例 主要 通过 设置 CardView 控件 的 android: foreground 属性 为 自 定 义 selector, 实 现在 按 下 
CardView 控件 时 新 增 线性 渐变 前 景 层 的 效果 。 当 实例 运行 之 后 , 单 击 CardView 控件 , 则 在 按 下 该 控 
件 时 将 在 该 控件 上 新 增 由 蓝 到 绿 的 线性 渐变 ( 半 ) 透 明 层 ,如 图 053. 1 的 左 图 所 示 ; 在 离开 该 控件 时 将 
移 除 该 层 ( 恢 复 ) 至 正常 状态 ,如 图 053. 1 的 右 图 所 示 。 
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MySample MySample 


图 053. 1 
主要 代码 如 下 : 


< android. support. v7. widget. CardView 
xmlns:card view= "http://schemas.android. com/apk/res — auto" 
android: layout_width = "200dp" 
android: layout height = "wrap_content" 
android:layout_centerInParent = "true" 
android:clickable = "true" 
android:foreground = "(Odrawable/myselector" 
card view:cardBackgroundColor = " # 00EEEE" 
card view:cardCornerRadius = "20dp" 
card view:cardElevation = "10dp" 
card view:cardPreventCornerOverlap = "true" 
card_view:cardUseCompatPadding = "true" 
card_ view:contentPadding = "15dp"> 
<TextView android:layout width= "wrap content" 
android:layout height = "wrap_content" 
android:text = "量子 通信 和 是 指 利用 量子 纠缠 效应 进行 信息 传递 的 一 种 新 型 的 通讯 方式 .量子 通 
信 主 要 涉及 :量子 密码 通信 ,量子 远程 传 态 和 量子 密集 编码 等 .高 效 安全 的 信息 传输 日 益 受 到 人 们 的 关注 ." 
android:textSize = "20dp"/> 
</android. support. v7. widget. CardView > 


上 面 这 段 代 码 在 MyCode\MySample574\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: foreground 二 "(drawable/myselector" 用 于 设置 具有 半 透 明 线 性 渐变 前 景 层 
的 自 定义 selector。 自 定义 selector 的 主要 代码 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< selector xmlns:android = "http://schemas. android. com/apk/res/android"> 
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< item android: state pressed = "true"> 
< Shape> < gradient android: startColor = " # AAOOOOFF" 
android:endColor = " # AAOOFF00" android:type = "linear" /> 
</ shape ></ item > 
< item> 
< shape>< solid android: color = "(Oandroid: color/transparent" /></shape> 


</ item></selector > 


上 面 这 段 代码 在 MyCode\MySample574\app\src\main\res\drawable\myselector. xml 文件 中 。 
在 这 段 代 人 码 中 ,< item android: state_pressed 二 "true"> 表 示 在 控件 被 按 下 时 添加 此 标签 中 的 线性 渐 
变 前 景 层 。android: startColor 一" # AAO000FF "表示 渐变 开始 颜色 是 半 和 透明 蓝 色 。android: 
endColor 王 " 井 AA0O0FF00" 表 示 渐 变 结 束 颜色 是 半 透 明 绿 色 。android: type 王 “linear" 表 示 渐 变 类 型 
是 线性 渐变 。 此 外 ,在 Android Studio 中 ,使 用 此 实例 的 相关 控件 需要 在 gradle 中 引入 compile 'com. 
android. support: cardview-v7: 25. 3.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample574 文件 
夹 中 。 


054 ”使 用 ViewSwitcher 平 请 切换 两 个 View 


此 实例 主要 通过 使 用 ViewSwitcher 控件 ,实现 以 动画 的 形式 在 两 个 View 之 间 进 行 平滑 切换 ， 
当 实 例 运行 之 后 , 单 击 “下 一 个 视图 ”按钮 , 则 将 显示 第 二 个 View, 如 图 054. 1 的 右 图 所 示 ; 再 次 单 击 
“下 一 个 视图 ”按钮 , 则 将 显示 第 一 个 View, 如 图 054. 1 的 左 图 所 示 。 如 果 在 实例 运行 之 后 , 单 击 “上 
一 个 视图 ”按钮 ,也 将 显示 第 二 个 View, 如 图 054. 1 的 右 图 所 示 ; 再 次 单 击 * 上 一 个 视图 ”按钮 , 则 将 显 
示 第 一 个 View ,如 图 054. 1 的 左 图 所 示 。 由 于 ViewSwitcher 控件 只 支持 两 个 View 的 切换 ,并 且 周 
而 复 始 地 循环 ,因此 两 个 按钮 的 功能 完全 相同 。 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

ViewSwitcher myViewSwitcher; 

Animation mySlideInLeft, mySlideOutRight; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myViewSwitcher = (ViewSwitcher) findViewById(R. id.viewswitcher); 
mySlideInLeft = AnimationUtils. loadAnimation(this, 


android. R.anim. slide in left); // 加 载 从 左边 滑 入 的 动画 
mySlideOutRight = AnimationUtils. loadAnimation(this, 
android. R. anim. slide out right); // 加 载 从 右边 滑 出 的 动画 


myViewSwitcher. setInAnimation(mySlideInLeft); 
myViewSwitcher. setOutAnimation(mySlideOutRight); 
} 
// 响 应 单 击 "上 一 个 视图 "按钮 
public void onClickmyBtnl (View v) { myViewSwitcher. showPrevious( );} 
// 响 应 单 击 " 下 一 个 视图 "按钮 
public void onClickmyBtn2(View v) { myViewSwitcher. showNext(); } 


上 面 这 段 代 码 在 MyCode\ MySample386 \app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,ViewSwitcher 是 一 个 视图 切换 控件 , 它 可 以 将 两 个 View 
放 在 一 起 ,每 次 只 显示 一 个 View。 当 从 一 个 View 切换 到 男 一 个 View 时 ,ViewSwitcher 支持 指定 的 
动画 效果 。ViewSwitcher 的 showPrevious() 方 法 用 于 显示 上 一 个 View。ViewSwitcher 的 showNext() 方 
法 用 于 显示 下 一 个 View。 在 XML 文件 中 ,ViewSwitcher 添加 View 的 主要 代码 如 下 : 


< ViewSwitcher android:id= "(0@ + id/viewswitcher" 
android: layout_width = "match parent" 
android: layout height = " match parent"> 
< ImageView android: layout width = " match parent" 
android: layout height = " match parent" 
android: scaleType = "fitXY" 
android: src = " (Qnmipmap/myimagel1" /> 
< ImageView android: scaleType = "fitXxY" 
android: layout width = "match parent" 
android: layout height = " match parent" 
android: src =" (QWmipmap/myimage2" /> 
</ViewSwitcher > 


上 面 这 段 代 码 在 MyCode\MySample386\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,< ImageView > 代表 一 个 View ,也 可 以 使 用 LinearLayout 等 布局 控件 创建 更 复杂 的 
View。 此 实例 的 完整 项 目 在 MyCode\MySample386 文件 夹 中 。 


055 ”使 用 SlidingDrawer 实现 抽 敢 式 滑动 


此 实例 主要 通过 使 用 SlidingDrawer 控件 ,在 垂直 方 回 实现 抽 屋 式 的 拖 电 滑 动 效 果 。 当 实例 运行 
之 后 , 按 住 底部 的 加 上 和 光 头 回 顶 部 移动 , 则 下 面 的 大 桥 图 像 也 跟随 回 上 移动 ,如 图 055. 1 的 左 图 所 示 ， 
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如 果 松 开 手 指 , 则 大 桥 图 像 将 自动 滑 回 顶部 , 回 上 箭头 变 为 回 下 箭头 。 当 大 桥 图 像 位 于 顶部 时 , 回 上 
季 头 自动 变 为 癌 下 箭头 ,此 时 拖 动 回 下 熏 头 回 底 部 移动 , 则 下 面 的 大 桥 图 像 也 跟随 回 底 部 移动 ,如 
图 055. 1 的 右 图 所 示 。 
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图 055. 1 
主要 代码 如 下 : 


< SlidingDrawer android: id = "@ + id/mySlidingDrawer" 
android:background = "(@mipmap/myimagel" 
android:layout width = "fill parent" 
android:layout height = "fill parent" 
android: content = "(0@ + id/myLinearLayout" 
android: handle = " (@ + id/myImageButton" 
android:orientation = "vertical"> 
< ImageButton android: id = " (Qid/myImageButton" 
android:layout width= "match _ parent” 
android: layout_ height = "wrap_content" 
android:background = " # O00FFFF" 
android: src = "(mipmap/buttonup"/> 
<LinearLayout android: id = " (Qid/myLinearLayout" 
android:background = "# ffffff" 
android:orientation = "vertical" 
android: layout_ width = "match parent" 
android:layout height = "match parent"> 
<TextView android:id= "(@ + id/myTextView" 
android:layout width= "match parent" 
android:layout height = "60dp" 
android:gravity = "center vertical |center horizontal" 
android:text = "滑动 抽 屋 " 
android:background = " # EE82EE" 
android:textColor =" 间 000000" 
android:textSize = "22dp" 
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android:textStyle = "bold"/> 

< ImageView android:id= "(@ + id/myImageView" 
android:layout width= "match parent" 
android:layout height = "wrap_content” 
android: scaleType = "fitXY" 
android:src = "(@mipmap/myimage2"/> 

</LinearLayout > 

</SlidingDrawer > 


上 上面 这 段 代 码 在 MyCode\MySample359\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,SlidingDrawer 控件 由 两 个 View 组 成 ,一 个 是 可 以 拖 动 的 handle, 一 个 是 隐藏 内 容 的 
View, 它 里 和 面 的 控件 必须 设置 布局 ,在 布局 文件 中 必须 指定 handle 和 content。 在 Java 代码 中 ,可 以 
通过 监听 器 获取 和 设置 SlidingDrawer 在 滑动 过 程 中 的 信息 ,主要 代码 如 下 : 


public class MainActivity extends Activity { 
private SlidingDrawer mySlidingDrawer; 
private ImageButton myImageButton; 
private Boolean myFlag = false; 
private TextView TextView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity main); 
myImageButton = (ImageButton)findViewById(R. id. myImageButton); 
mySlidingDrawer = (SlidingDrawer)findViewById(R. id.mySlidingDrawer); 
TextView = (TextView)findViewById(R. id. myTextView); 
mySlidingDrawer. setOnDrawerOpenListener( 
new SlidingDrawer. OnDrawerOpenListener( ){ 
(@ Override 
public void onDrawerOpened( ) { 
myFlag = true; 
myImageButton. setImageResource(R. mipmap. buttondown); 
| 忆 2 
mySlidingDrawer. setOnDrawerCloseListener( 
new SlidingDrawer. OnDrawerCloseListener( ){ 
(Override 
public void onDrawerClosed() { 
myFlag = false; 
myImageButton. setImageResource(R. mipmap. buttonup); 
Ps 
mySlidingDrawer. setOnDrawerScrollListener( 
new SlidingDrawer. OnDrawerScrollListener( ){ 
(Override 
public void onScrollEnded() { TextView. setText(" 结束 拖 动 "); } 
@ Override 
public void onScrollStarted( ) { TextView. setText(" 开始 拖 动 "); } 
Ff 本 要 


上 面 这 段 代 码 在 MyCode\ MySample359\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample359 文件 夹 中 。 
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056 ” 自 定 义 ScrollView 实现 下 拉 回 弹 动 画 


此 实例 主要 通过 在 自 定义 ScrollView 的 onLayout 事件 啊 应 方法 中 记录 其 子 视图 (控件 ) 的 初始 
位 置 ,然后 重 写 dispatchTouchEvent 事件 啊 应 方法 来 监听 手势 滑动 操作 ,并 在 重 置 子 视 图 (控件 ) 位 置 
时 为 其 添加 平移 动画 ,从 而 实现 下 拉 回 弹 效 果 。 当 实例 运行 之 后 ,手指 按 住 图 像 回 下 滑动 , 则 图 像 随 
之 回 下 滑动 ,如 图 056. 1 的 左 图 所 示 ; 然后 松 开 手指 , 则 图 像 回 弹 到 初始 位 置 ( 即 顶端 ), 如 图 056. 1 的 
右 图 所 示 。 


MySample 


图 056. 1 
主要 代码 如 下 : 


< com. bin. luo. mysample. MyScrollView 
android: layout width= "match parent" 
android:layout height = "match parent"> 
< ImageView 
android:layout width= "match parent" 
android:layout height = "match parent" 
android: scaleType = "fitXY" 
android:src = "(@mipmap/myimagel"/> 
</com. bin. luo. mysample. MyScrollView > 


上 面 这 段 代码 在 MyCode\MySample709\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,com. bin. luo. mysample. MyScrollView 表示 自 定 义 控件 MyScrollView ,实现 下 拉 回 
弹 的 ImageView 控件 应 该 放置 在 其 里 面 。MyScrollView 的 自 定义 代码 如 下 : 


public class MyScrollView extends ScrollView { 


// 拉 出 屏幕 时 的 拖 忠 系数 , 如果 是 1, 类似 拖 动 效 果 
private static final float MOVE DELAY = 0.3f; 
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private static final int ANIM TIME = 300; // 回 弹 动画 持续 时 间 
private View myChildView; 
private boolean bMoved; 
private Rect myOriginalView = new Rect(); 
private float startY; 
public MyScrollView(Context context, AttributeSet attrs, int defStyle) { 
super(context, attrs, defStyle); 
} 
public MyScrollView(Context context, AttributeSet attrs){super(context, attrs); } 
public MyScrollView(Context context) {super(context);} 
(Override 
protected void onFinishInflate() { 
super. onFinishInflatel( ) ; 
if (getChildCount() > 0) { myChildView = getChildAt(0); } 
} 
(Override 
protected void onLayout(boolean changed，int 1, int 七 ，int r, int b) { 
super. onLayout (changed, 1, t, r, b); 
if (myChildView == null) return; 
myOriginalView. set(myChildView. getLeft(), myChildView. getTop() ， 
myChildView.getRight(), myChildView. getBottom( ) ) ; 
} 
(Override 
public boolean dispatchTouchEvent(MotionEvent ev) { 
if (myChildView == null) { return super. dispatchTouchEvent(ev); } 
int myAction = ev.getRction( ) ; 
Switch (myAction) { 
case MotionEvent. ACTION DOWN: 
startY = ev.getY(); 
break; 
case MotionEvent. ACTION UP: 
case MotionEvent. ACTION CANCEL: 
if (!bMoved) break; 
TranslateAnimation myAnim = new TranslateAnimation(0, 0, 
myChildView. getTop( ), myOriginalView. top); // 执 行 回 弹 动画 
myAnim. setDuration(ANIM TIME); 
myChildView. startAnimation(myAnim); 
bMoved = false; 
resetViewLayout( ); 
break; 
case MotionEvent. ACTION MOVE: 
float nowY = ev.getY(); 
int deltaY = (int) (nowY - startY); 
int offset = (int) (deltaY * MOVE DELAY); 
myChildView. layout (myOriginalView. left, myOriginalView.top + offset, 
myOriginalView.right, myOriginalView. bottom + offset); 
bMoved = true; 
break; 
default: 


return super.dispatchTouchEvent(ev); 
} 
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public void resetViewLayout() { 
myChildView. layout( myOriginalView. left, 
myOriginalView. top, 
myOriginalView. right., 
myOriginalView. bottom) ; 
半 ， 


上 F 面 这 段 代 码 在 MyCode\ MySample709\app\src\ main\java\com\bin\luo\mysample\ 
MyScrollView. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample709 文件 夹 中 。 


057 ”使 用 CollapsingToolbarLayout 实现 深 动 折 炙 


此 实例 主要 通过 使 用 AppBarLayout. LayoutParams 的 setScrollFlags() 方 法 设置 滚动 标志 ,实现 
在 CollapsingToolbarLayout 中 的 控件 出 现 滚 动 折 县 效果 。 当 实例 运行 之 后 , 回 下 拖 动 工具 栏 , 则 工 
具 栏 图 像 展 开 , 如 图 057. 1 的 左 图 所 示 ; 回 上 拖 动 工具 栏 , 则 工具 栏 图 像 折 二 , 直 到 图 像 消 失 , 如 
图 057. 1 的 右 图 所 示 。 
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图 057. 1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
CollapsingToolbarLayout myCollapsingToolbarLayout = (CollapsingToolbarLayout) 
findViewById(R. id.myCollapsingToolbarLayout ); 
AppBarLayout. LayoutParams myLayoutParams = (AppBarLayout. LayoutParams ) 
myCollapsingToolbarLayout. getLayoutParams( ); 
// 启 用 滚动 折 释 功能 
myLayoutParams. setScrollFlags( 
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AppBarLayout. LayoutParams. SCROLL FLAG SCROLL| 
AppBarLayout. LayoutParams. SCROLL FLAG EXIT UNTIL COLLAPSED); 
WebView myWebView = (WebView)findViewById(R. id. myWebView); 
myWebView. getSettings( ). setJavaScriptEnabled(true) ; 
myWebView. setWebViewClient(new WebViewClient( ) ) ; 
myWebView. loadUrl("http://www. baidu. com" ); 
} 
} 


上 面 这 上段 代码 在 MyCode\ MySample910\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myLayoutParams. setScrollFlags (AppBarLayout. 
LayoutParams. SCROLL FLAG SCROLL|AppBarLayout. LayoutParams. SCROLL FLAG EXIT_ 
UNTIL_ COLLAPSED) 用 于 实现 CollapsingToolbarLayout 的 滚动 折 和 县 功能 , 它 实际 上 等 效 于 布局 文 
件 (MyCode\MySample910\app\src\main\res\layout\activity_main. xml) 的 属性 设置 : app: layout 
scrollFlags 二 ”scroll | exitUntilCollapsed "。 需 要 说 明 的 是 ,在 Android Studio 中 ,使 用 
CollapsingToolbarLayout 控件 需要 在 gradle 中 引入 compile 'com. android. support: design: 25. 0. 1， 
依赖 项 。 男 外 ,如 果 应 用 的 主题 是 android: Theme. Material. Light. DarkActionBar, 可 能 在 运行 时 会 
衣 江 ,因此 需要 将 app\src\main\res\ values\styles. xml 文件 的 主题 < style name = 二" AppTheme" 
parent 二 "android: Theme. Material. Light. DarkActionBar"> 修 改 为 < style name = 二" AppTheme" 
parent 二 ”Theme. AppCompat. Light. DarkActionBar">。 此 实例 的 完整 项 目 在 MyCode\ 
MySample910 文件 夹 中 。 


058 ”使 用 BottomNavigationView 实现 底部 导航 


此 实例 主要 演示 了 使 用 BottomNavigationView 控件 ,在 屏幕 底部 实现 菜单 风格 的 导航 效果 。 当 
实例 运行 之 后 ,在 屏幕 底部 将 显示 5 个 导航 菜单 项 ,并 且 每 个 菜单 项 同时 显示 图 标 和 文本 ; 单 击 第 1 
个 菜单 项 ,将 显示 该 菜单 项 对 应 的 图 像 , 如 图 058. 1 的 左 图 所 示 ; 单 击 第 5 个 菜单 项 ,也 将 显示 该 菜单 
项 对 应 的 图 像 ,如 图 058. 1 的 右 图 所 示 ; 单 击 其 他 菜单 项 将 实现 类 似 的 功能 。 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
BottomNavigationView myBottomNavigationView; 
ImageView myImageView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
myBottomNavigationView = 
(BottomNavigationView) findViewById(R. id.myBottomNavigationView); 
myBottomNavigationView. setOnNavigationItemSelectedListener( 
new BottomNavigationView. OnNavigationItemSelectedListener() { 
(Override 
public boolean onNavigationItemSelected(MenuItem item) { 
Switch (item. getItemId()) { 
case R. id. myIteml : 
myImageView. setImageResource(R. mipmap.myimagel); 
break; 
case R. id. myItem2: 
myImageView. setImageResource(R. mipmap. myimage2 ) ; 
break; 
case R. id.myItem3: 
myImageView. setImageResource(R. mipmap. myimage3 ); 
break; 
case R. id.myItem4: 
myImageView. setImageResource(R. mipmap. myimage4 ); 
break; 
case R. id.myItem5: 
myImageView. setImageResource(R. mipmap. myimage5 ); 
break; 


避 太 ? 几 半 ， 


上 面 这 段 代 码 在 MyCode\ MySample433\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,onNavigationItemSelected() 方 法 主要 用 于 啊 应 用 户 单 击 
在 导航 栏 中 的 菜单 项 , 即 确定 用 户 单 击 了 导航 栏 的 哪 一 个 菜单 项 。 导 航 栏 的 菜单 项 设置 则 是 在 布局 
文件 中 实现 的 ,主要 代码 如 下 : 


<LinearLayout android:layout width= "match parent" 
android:layout_ height = "match parent" 
android:orientation = "vertical"> 
< ImageView android:id = "(@ + id/myImageView" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android: layout weight = "10" 
android: scaleType = "fitXY" 
android:src = "(Qnmipmap/myimagel"/> 
< android. support. design. widget. BottomNavigationView 
android: id = " (@ + id/myBottomNavigationView" 
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android: layout width = "match parent" 

android: layout height = "56dp" 

android: layout alignParentBottom = "true" 

android: layout weight ="1" 

android: background = " # 00FFFF" 

app: itemIconTint = " # FF0000" 

app: itemTextColor = " (Wandroid: color/black" 

app: menu = " (Wmenu/mynavigation" /> 
</LinearLayout > 


上 面 这 段 代 码 在 MyCode\MySample433\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,app: menu 二 "(menu/mynavigation" 的 属性 值 mynavigation 即 是 导航 菜单 布局 文件 ; 
如 果 在 res 目录 中 没有 menu 目录 ,请 首先 创建 menu 目录 ,然后 再 添加 mynavigation. xml 文件 。 
mynavigation. xml 文件 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 
< menu xmlns:android = "http://schemas. android. com/apk/res/android"> 
< item 
android: id= "(@ + id/myIteml1" 
android: icon = "(@mipmap/mymovie" 
android:title = "我 是 传奇 "/> 
< item 
android: id= "@ + id/myItem2" 
android: icon = "(@mipmap/mymovie" 
android:title = " 银 辟 杀 手 "/> 
< item 
android:id= "(@ + id/myItem3" 
android: icon = "(@mipmap/mymovie" 
android:title= "三 百 斯 巴 达 勇 士 "/> 
< item 
android:id= "@ + id/myItem4" 
android: icon = "(@mipmap/mymovie" 
android:title= "功夫 熊猫 "/> 
< item 
android:id= "(@ + id/myItem5" 
android: icon = "(@mipmap/mymovie" 
android:title= "怒火 救援 "/> 
</menu > 


上 面 这 段 代 码 在 MyCode\MySample433\app\src\main\res\menu\mynavigation. xml 文件 中 。 
需要 说 明 的 是 ,在 Android Studio 中 使 用 BottomNavigationView 控件 需要 在 gradle 中 引入 compile 
'com. android. support: design: 25. 0.0' 依 赖 项 ,25 是 开发 环境 的 SDK 版 本 号 。 男 外 ,如 果 应 用 的 主 
题 是 android: Theme. Material. Light. DarkActionBar, 应 用 可 能 在 运行 时 会 月 溃 ,因此 需要 将 app\ 
src\main\res\values\styles. xml 文件 的 主题 < style name 一 "AppTheme" parent 二 "android: Theme. 
Material. Light. DarkActionBar"> 修 改 为 < style name 王 "AppTheme"”parent 一 "Theme. AppCompat. 
Light. DarkActionBar">。 此 实例 的 完整 项 目 在 MyCode\MySample433 文件 夹 中 ， 


059 ”在 ProgressBar 上 同时 显示 两 种 进度 


此 实例 主要 通过 使 用 setSecondaryProgress() 方 法 设置 第 二 进度 值 ,实现 在 ProgressBar 控件 上 
同时 显示 两 种 进度 。 当 实例 运行 之 后 , 单 击 “ 增 加 进度 ”按钮 , 则 在 ProgressBar 控件 上 第 一 进度 ( 青 
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色 ) 和 第 二 进度 (红色 ) 将 同时 增加 10%; 单 击 “ 减 少 进度 ”按钮 , 则 在 ProgressBar 控件 上 第 一 进度 和 
第 二 进度 将 同时 减少 10% ,效果 分 别 如 图 059. 1 的 左 图 和 右 图 所 示 。 需 要 注意 的 是 : 如 果 第 二 进度 
达到 100% ,第 一 进度 未 达到 100% , 则 单 击 “增加 进度 ”按钮 仅 增 加 第 一 进度 。 


DJ rg ( 上 午 8:4 ¢€9 Onepa 2 Mpenion 


MySample MySample 


增加 进度 减少 进度 增加 进度 减少 进度 


第 一 进度 : 20.0% 第 二 进度 : 50.0% 第 一 进度 : 10.0% 第 二 进度 : 40.0% 


图 059. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
TextView myTextView; 
ProgressBar myProgressBar; 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myTextView = (TextView) findViewById(R. id.myTextView); 
myProgressBar = (ProgressBar) findViewById(R. id.myProgressBar); 
// 设 置 第 二 进度 的 显示 颜色 
myProgressBar. setSecondaryProgressTintList( 
getResources( ) . getColorStateList (android.R.color.holo red dark)); 
myTextView. setText(" 第 一 进度 :" + myProgressBar. getProgress() * 100. 0/ myProgressBar. getMax( ) + 
"$"+ "第 二 进度 :" + myProgressBar.getSecondaryProgress() * 100.0/ myProgressBar. getMax() + "%"); 
} 
// 响 应 单 击 按钮 "增加 进度 " 
public void onClickBtnl (View v) { 
myProgressBar. incrementProgressBy(10); 
myProgressBar. setSecondaryProgress(myProgressBar. getProgress() + 30); 
myTextView. setText(" 第 一 进度 :" + myProgressBar. getProgress() * 100.0 / myProgressBar.getMax() + 
"%g"” +" 第 二 进度 :" + myProgressBar. getSecondaryProgress() * 100. 0 / myProgressBar. getMax ( ) 
wa- 
} 
// 啊 应 单 击 按钮 "减少 进度 " 
public void onClickBtn2(View v) { 
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myProgressBar. incrementProgressBy( — 10); 

myProgressBar. setSecondaryProgress(myProgressBar. getProgress() + 30); 

myTextView. setText(" 第 一 进度 :" + myProgressBar. getProgress() * 100.0 / myProgressBar.getMax() + 
"%g"” +" 第 二 进度 :" + myProgressBar. getSecondaryProgress() * 100. 0 / myProgressBar. getMax ( ) 
下 


风 


上 面 这 段 代 码 在 MyCode\MySample797\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 , myProgressBar. incrementProgressBy (10) 用 于 设置 第 一 进度 增 量 ， 
myProgressBar. setSecondaryProgress (myProgressBar. getProgress ( ) 十 30) 用 于 设置 第 二 进度 ， 
myProgressBar. getProgress() 用 于 获取 第 一 进度 的 当前 值 ,myProgressBar. getSecondaryProgress() 用 于 获 
取 第 二 进度 的 当前 值 。 此 实例 的 完整 项 目 在 MyCode\MySample797 文件 夹 中 。 


060 ”使 用 YiewOutlineProvider 创建 圆 角 控 件 


此 实例 主要 通过 创建 圆 角 和 矩形 风格 的 ViewOutlineProvider 自 定 义 样式 , 并 使 用 
setO 〇 utlineProvider() 方 法 和 setClipToOutline() 方 法 ,从 而 实现 为 ImageView 控件 添加 圆 角 效果 。 
当 实 例 运 行 之 后 , 单 击 “设置 圆 角 效果 ”按钮 , 则 ImageView 控件 (电影 海报 图 像 ) 的 圆 角 效 果 如 
图 060. 1 的 左 图 所 示 ; 单 击 “取消 圆 角 效果 ?按钮 , 则 ImageView 控件 (电影 海报 图 像 ) 的 正常 显示 效 
果 如 图 060. 1 的 右 图 所 示 。 


MySample MySample 


设置 圆 角 效果 取消 圆 角 效果 设置 圆 角 效果 取消 圆 角 效果 


图 060. 1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
ImageView myImageView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 


super. onCreate( savedInstanceState) ; 
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setContentView(R. layout. activity main); 
myImageView = ( ImageView)findViewById(R. id. myImageView) ; 
ViewOutlineProvider myProvider = new ViewOutlineProvider() { 
public void getOutline(View view, Outline outline) { 
outline. setRoundRect(0, 0, view.getWidth( ), 


view. getHeight(), 60); //60 表示 圆 角 半径 
} }; 
myImageView. setOutlineProvider(myProvider); 
} 
public void onClickmyBtnl (View v) { // 响 应 单 击 " 设 置 圆 角 效 果 " 按 钮 
myImageView. setClipToOutline(true); 
} 
public void onClickmyBtn2 (View v) { // 响 应 单 击 "取消 圆 角 效 果 " 按 钮 
myImageView. setClipToOutline(false); 
} } 


上 面 这 段 代 码 在 MyCode\ MySample558\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myProvider 三 new ViewOutlineProvider(){} 用 于 创建 一 
个 圆 角 半径 为 60 像素 的 ViewOutlineProvider。myImageView。setOutlineProvider(myProvider) 表 
示 myImageView 控件 使 用 myProvider 的 圆 角 形状 。myImageView. setClipToOutline (true) 表示 使 用 
myProvider 裁剪 myImageVievw 控件 的 外 形 。 此 实例 的 完整 项 目 在 MyCode\MySample558 文件 夹 中 。 


061 使 用 AnalogClock 创建 自 定义 时 钟 


此 实例 主要 通过 设置 AnalogClock 控件 的 android: dial 和 android: hand_minute 属性 ,并 使 用 
Timer 中 的 schedule() 方 法 定时 刷新 时 间 ,创建 一 个 动态 走动 的 时 钟 。 当 实例 运行 之 后 ,当时 间 走 到 
11: 01 ,时 钟 的 显示 效果 如 图 061. 1 的 左 图 所 示 ; 当时 间 走 到 11: 04, 时 钟 的 显示 效果 如 图 061. 1 的 右 图 


所 示 。 


MySample MySample 


图 061. 1 
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< AnalogClock android:id= "@ + id/myAnalogClock" 
android: layout width= "wrap_content" 
android: layout_ height = "wrap_content" 
android: dial = " (mipmap/mywatch" 
android: hand minute = " (mipmap/myhand" /> 


上 面 这 段 代 码 在 MyCode\MySample055\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: dial = 二" (@ mipmap/mywatch" 表示 AnalogClock 控件 的 表盘 dial 采用 
mywatch 资源 进行 配置 。android: hand_minute 王 "mipmap/ymyhand" 表 示 AnalogClock 控件 的 分 
针 hand_minute 采用 myhand 资源 进行 配置 。AnalogClock 控件 的 时 间 走 动 效 果 则 通过 Java 代码 实 
现 , 主 要 代码 如 下 : 


public class MainActivity extends Activity { 
private AnalogClock myAnalogClock; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity main); 
myAnalogClock = (AnalogClock) findViewById(R. id.myAnalogClock); 
new Timer( ) . schedule(new TimerTask( ) { // 定 义 周期 性 地 改变 时 钟 的 指针 
@Override 
public void run() { myAnalogClock. postInvalidate( ); } 
}, 0, 1000); } } 


上 面 这 段 代 码 在 MyCode\ MySample055\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,new Timer(). schedule(new TimerTask() { (@Override 
public void run ( ) { myAnalogClock. postInvalidate (); ; ),，0,1000) 表 示 每 阳 1000ms 调用 
postInvalidate() 方 法 更 新 myAnalogClock 一 次 ,中 间 无 延迟 (0 表示 无 延迟 )。 此 实例 的 完整 项 目 在 
MyCode\MySample055 文件 夹 中 ，。 


062 ”在 TextClock 中 定制 日 期 格式 


此 实例 主要 通过 设置 TextClock 控件 的 format12Hour 属性 ,并 使 用 Timer 的 schedule() 方 法 定 
时 刷新 时 间 ,实现 以 指定 的 格式 动态 显示 时 间 信 息 。 当 实例 运行 之 后 ,将 以 指定 的 格式 动态 显示 当前 
系统 的 时 间 信 息 ,如 图 062. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


< TextClock android: id = "(@ + id/myTextClock" 
android:layout width= "wrap_content" 
android:layout height = "wrap_content" 
android:textSize = "18dp"” 
android:textColor = "#00f" 
android:format12Hour = "yyyyY 年 MM 月 dd 日 a hh:mm: ss EEEE" /> 


上 面 这 段 代 码 在 MyCode\MySample056\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: formatl2Hour 二 "yyyy 年 MM 月 dd 日 a hh: mm:ssEEEE" 中 的 yyyy 表示 
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dx 要 ”| 日 98% 上 午 11:16 


2018 年 06 月 08 日 上 午 11:15:49 星期 五 2018 年 06 月 08 日 上 午 11:16:06 星期 五 


图 062.1 


年 数 、.MM 表示 月 数 、dd 表示 日 数 、hh 表示 小 时 数 .mm 表示 分 钟 数 、ss 表示 秒 数 、EEEE 表示 星期 数 、 
a 表示 上 /下 午 。TextClock 控件 的 时 间 动 态 显 示 效 果 则 通过 Java 代码 实现 ,主要 代码 如 下 : 


public class MainActivity extends Activity { 
private TextClock myTextClock.; 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
myTextClock = (TextClock) findViewById(R. id.myTextClock); 


new Timer( ) . schedule(new TimerTask( ) { // 周 期 性 地 改变 时 钟 的 指针 
(Override 
public void run( ) { myTextClock. postInvalidate(); } }, 0, 1000); 

Ey 


上 和 面 这 段 代 码 在 MyCode\ MySample056\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample056 文件 夹 中 ， 


063 ”使 用 RatingBar 实现 星 级 评分 


此 实例 主要 通过 使 用 RatingBar 控件 ,实现 星 级 评分 的 功能 。 当 实例 运行 之 后 , 当 在 RatingBar 
控件 中 选择 不 同位 置 的 五 星 时 ,效果 分 别 如 图 063. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 

public class MainActivity extends Activity { 


(Override 
protected void onCreate(Bundle savedInstanceState) { 
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super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
final ImageView myImageView = (ImageView) findViewById(R. id. myImageView) ; 
RatingBar myRatingBar = (RatingBar) findViewById(R. id. myRatingBar) ; 
myRatingBar. setOnRatingBarChangeListener( 

new RatingBar. OnRatingBarChangeListener() { 


@ Override // 当 选择 不 同位 置 的 五 星 时 触发 该 方法 
public void onRatingChanged(RatingBar arg0, float rating, boolean fromUser) { 


// 动 态 改变 图 像 透明 度 ,255 表示 星 级 评分 条 最 大 值 , 5 个 星星 就 代表 最 大 值 255 
myImageView. setAlpha( (int) (rating * 255 / 5)); 
Be] 
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图 063. 1 


上 面 这 段 代 码 在 MyCode\ MySample284 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,setOnRatingBarChangeListener() 事 件 啊 应 方法 用 于 啊 应 
用 户 在 RatingBar 控件 上 选择 不 同位 置 的 五 星 ,RatingBar 控件 的 用 法 和 功能 与 SeekBar 控件 十 分 相 
似 , 它 们 都 允许 用 户 通过 拖 动 来 改变 进度 ,RatingBar 与 SeekBar 的 最 大 区 别 在 于 ; RatingBar 使 用 五 
星 表 示 进 度 。 此 实例 的 完整 项 目 在 MyCode\MySample284 文件 夹 中 。 


064 ”在 登录 窗口 中 使 用 SeekBar 实现 手动 校 验 


此 实例 主要 通过 比较 SeekBar 滑 块 的 最 大 值 ,实现 在 登录 窗口 强制 手动 操作 校 验 。 当 实例 运行 
之 后 ,在 “账户 名 称 : ”和 “账户 密码 : ”输入 框 中 输入 内 容 ,然后 单 击 “ 登 录 ” 按 钮 ,此 时 “登录 ”按钮 没有 
任何 反应 ,如 图 064. 1 的 左 图 所 示 ; 如 果 根 据 提示 “请 按 住 滑 块 , 拖 动 到 最 右边 ”进行 操作 , 则 显示 “ 验 
证 成 功 ”, 然 后 自动 消失 ,如 图 064. 1 的 右 图 所 示 , 此 时 单 击 “ 登 录 ” 按 钮 , 则 会 弹出 Toast。 
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MySample MySample 


账户 名 称 : binluobin 账户 名 称 : binluobin 


账户 密码 : 本 账户 密码 : 和 


登录 


图 064. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements SeekBar. OnSeekBarChangeListener { 
public Handler myHandler = new Handler() { 
(QOverride 
public void handleMessage(Message msg) { 
super. handleMessage(msg); 
if (msg.what == 1) { 
mySeekBar. setVisibility(View. GONE); 
myTextView. setVisibility(View. GONE); 
} 
myBtnLogin. setOnClickListener(new View. OnClickListener() { 
(Override 
public void onClick(View view) { 
Toast. makeText(MainActivity.this, "登录 成 功 !" ，Toast.LENGTH SHORT). show( ); 
finish(); } }); } }); 
private TextView myTextView; 
private SeekBar mySeekBar; 
private Button myBtnLogin; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myTextView = (TextView) findViewById(R. id.myTextView); 
mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
mySeekBar. setOnSeekBarChangeListener(this); 
myBtnLogin = (Button) findViewById(R. id.myBtnLogin); 
myBtnLogin. setClickablel( false); 
} 
(Override 
public void onProgressChanged( SeekBar seekBar, int i, boolean b) { 
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// 如 果 滑 块 抵 达 右 端 最 大 值 , 则 设置 "登录 "按钮 为 可 用 状态 (之 前 该 按钮 是 不 可 用 的 ) 
if (seekBar.getProgress() == SeekBar. getMax()) { 
myTextView. setVisibility(View. VISIBLE); 
myTextView. setTextColor(Color. WHITE); 
myTextView. setText ("验证 成 功 !"); 
myBtnLogin. setClickable(true); 
new Thread() { 
(Override 
public void run() { 
super. run( ); 
try { 
Thread. sleep(1000); 
myHandler. sendEmptyMessage(1); 
} catch (InterruptedException e) { e.printStackTrace( ); } 
} }. start(); 
} else { myTextView. setVisibility(View. INVISIBLE); } 
} 
Override 
public void onStartTrackingTouch(SeekBar seekBar) { } 
(Override 
public void onStopTrackingTouch(SeekBar seekBar) { 
// 如 果 停 止 滑 动 滑 块 ,并 且 没 有 抵达 最 大 值 , 则 重 置 为 最 小 值 
if (seekBar.getProgress() != seekBar.getMax()) { 
seekBar. setProgress(0); 
myTextView. setVisibility(View. VISIBLE); 
myTextView. setTextColor(Color. GRAY); 
myTextView. setText(" 向 右 滑动 , 完成 验证 " ); 
i 
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上 和 面 这 上段 代码 在 MyCode\ MySample706\app\src\ main\java\com\bin\luo\ mysample\ 


MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample706 文件 夹 中 。 


文字 


065 ”使 用 ScaleXSspan 创建 局 平 风 格 的 文字 


此 实例 主要 通过 使 用 ScaleXSpan ,实现 在 TextView 中 以 扁平 化 的 风格 显示 文本 。 当 实例 运行 
之 后 ,在 TextView 中 以 遍 平 化 的 风格 显示 文本 的 效果 如 图 065. 1 所 示 。 
中 司 草 QH lB 944% 下 午 2:36 
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图 065. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
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TextView myTextView = (TextView) findViewById(R. id.myTextView); 
// 创 建 一 个 SpannableString 对 象 
SpannableString mySpannableString = new SpannableString(" 实例 集锦 " ); 
// 创 建 水 平 拉 伸 3 倍 的 ScaleXSpan 
mySpannableString. setSpan(new ScaleXSpan(3), 0, 
mySpannableString. length(), Spanned. SPAN EXCLUSIVE EXCLUSIVE); 

myTextView. setText(mySpannableString); 

}} 


上 和 面 这 上段 代码 在 MyCode\ MySample040\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString. setSpan (new ScaleXSpan(3)，0， 
mySpannableString. length(), Spanned. SPAN_EXCLUSIVE _ EXCLUSIVE) 表 示 将 指定 的 文字 在 水 
平方 回 拉 伸 3 倍 ,0 和 mySpannableString. length() 表 示 ScaleXSpan 作用 范围 的 起 止 位 置 。 此 实例 
的 完整 项 目 在 MyCode\MySample040 文件 夹 中 。 


066 ”使 用 MaskFilterSpan 实现 文字 边缘 模糊 


此 实例 主要 通过 定制 模糊 滤 镜 BlurMaskFilter 的 模糊 方式 为 BlurMaskFilter. Blur.。SOLID ,在 
TextView 中 实现 文本 的 背景 模糊 扩散 。 当 实例 运行 之 后 ,在 TextView 中 文本 的 背景 模糊 扩散 的 效 
果 如 图 066. 1 所 示 。 


MySample 


图 066.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(@Override 
protected void onCreate( Bundle savedInstanceState) { 
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super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableStringBuilder mySpannableStringBuilder = 
new SpannableStringBuilder(" 炫 酷 "); 
// 定 制 滤 镜 的 方式 是 BlurMaskFilter. Blur. SOLID 
MaskFilterSpan myMaskFilterSpan = new MaskFilterSpan( 
new BlurMaskFilter(21, BlurMaskFilter.Blur. SOLID)); 
mySpannableStringBuilder. setSpan(myMaskFilterSpan, 0, 2, 
Spanned. SPAN INCLUSIVE INCLUSIVE); // 设 置 滤 镜 的 作用 范 臣 
myTextView. setText(mySpannableStringBuilder); 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample037\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myMaskFilterSpan 一 new MaskFilterSpan (new 
BlurMaskFilter(21，BlurMaskFilter. Blur. SOLID)) 用 于 创建 一 个 模糊 距离 是 21, 模 糊 方 式 是 
BlurMaskFilter. Blur. SOLID 的 模糊 滤 镜 。mySpannableStringBuilder. setSpan (myMaskFilterSpan， 
0, 2，Spanned.SPAN _INCLUSIVE INCLUSIVE) 中 的 0 表示 模糊 滤 镜 在 字符 串 中 发 生 作 用 的 开始 
位 置 索 引 ,2 表示 模糊 滤 镜 在 字符 串 中 发 生 作 用 的 结束 位 置 索引 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample037 文件 夹 中 。 


067 ”使 用 MaskFilterSpan 实现 文字 中 心 铁 空 


此 实例 主要 通过 定制 模糊 滤 镜 BlurMaskFilter 的 模糊 方式 为 BlurMaskFilter. Blur. OUTER , 实 
现在 TextView 中 显示 线条 撒 边 的 铁 空 文本 。 当 实例 运行 之 后 ,在 TextView 中 显示 的 线条 描 边 的 铂 
空 文本 的 效果 如 图 067. 1 所 示 。 


MySample 


图 067.1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
TextView myTextView = (TextView) findViewById(R. id. myTextView) ; 
SpannableStringBuilder mySpannableStringBuilder = 
new SpannableStringBuilder(" 炫 酷 "); 
// 定 制 滤 镜 的 方式 是 BlurMaskFilter. Blur. OUTER 
MaskFilterSpan myMaskFilterSpan = new MaskFilterSpan( 
new BlurMaskFilter(1, BlurMaskFilter.Blur. OUTER)); 
mySpannableStringBuilder. setSpan(myMaskFilterSpan, 0, 2, 
Spanned. SPAN INCLUSIVE INCLUSIVE); // 设 置 滤 镜 的 作用 范围 
myTextView. setText(mySpannableStringBuilder); 
Bi 


上 面 这 段 代码 在 MyCode\MySample036\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 , myMaskFilterSpan 一 new MaskFilterSpan (new BlurMaskFilter (1， 
BlurMaskFilter. Blur. OUTER)) 用 于 创建 一 个 模糊 距离 是 1 ,模糊 方式 是 BlurMaskFilter. Blur. OUTER 的 
模糊 滤 镜 。mySpannableStringBuilder. setSpan (myMaskFilterSpan, 0, 2, Spanned. SPAN_INCLUSIVE 
INCLUSIVE) 中 的 0 表示 模糊 滤 镜 在 字符 串 中 发 生 作 用 的 开始 位 置 索 引 ,2 表示 模糊 滤 镜 在 字符 串 中 发 
生 作 用 的 结束 位 置 索 引 。 此 实例 的 完整 项 目 在 MyCode\MySample036 文件 夹 中 。 


068 ”使 用 MaskFilterSpan 实现 文字 整体 模糊 


此 实例 主要 通过 使 用 BlurMaskFilter, 实 现在 TextView 中 模糊 显示 文本 。 当 实例 运行 之 后 ,在 
TextView 中 模糊 显示 文本 的 效果 如 图 068. 1 所 示 。 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableStringBuilder mySpannableStringBuilder = 

new SpannableStringBuilder(" 炫 酷 " ); 
MaskFilterSpan myMaskFilterSpan = new MaskFilterSpan( 
new BlurMaskFilter(15, BlurMaskFilter.Blur. NORMAL)); 


mySpannableStringBuilder. setSpan(myMaskFilterSpan, 
0, 2, Spanned. SPAN INCLUSIVE INCLUSIVE); 


myTextView. setText (mySpannableStringBuilder); 
上 上] 


F 面 这 段 代 码 在 MyCode\ MySample035\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myMaskFilterSpan 一 new MaskFilterSpan (new 
BlurMaskFilter(15 ，BlurMaskFilter. Blur. NORMAL) ) 用 于 设置 模糊 滤 镜 的 模糊 距离 是 15 ,模糊 方 
式 是 BlurMaskFilter. Blur. NORMAL ,除了 NORMAL 之 外 ,常用 的 模糊 方式 还 有 : OUTER、 
INNER.\ SOLID, mySpannableStringBuilder. setSpan (myMaskFilterSpan., 0, 2, Spanned. SPAN _ 
INCLUSIVE_INCLUSIVE) 中 的 0 表示 模糊 滤 镜 在 字符 串 中 发 生 作 用 的 开始 位 置 案 引 ,2 表示 模糊 
滤 镜 在 字符 串 中 发 生 作 用 的 结束 位 置 索 引 。 此 实例 的 完整 项 目 在 MyCode\ MySample035 文件 夹 中 。 


069 ”使 用 MaskFilterSpan 模糊 多 个 字符 串 


此 实例 主要 通过 使 用 MaskFilterSpan, 实 现 非 连 续 的 多 个 字符 串 以 模糊 效果 显示 。 当 实例 运行 
之 后 ,在 “关键 词 : "输入 框 中 输入 “软件 ”, 然 后 单 击 “ 开 始 搜 索 ” 按 钮 , 则 在 下 面 的 文本 中 ,所 有 非 “ 软 
件 ” 的 文本 都 将 以 模糊 效果 显示 ,如 图 069. 1 的 左 图 所 示 。 在 "关键 词 : “输入 框 中 输入 “的 ”, 然 后 单 击 
“开始 搜索 ”按钮 , 则 在 下 面 的 文本 中 ,所 有 非 “ 的 ”的 文本 都 将 以 模糊 效果 显示 ,如 图 069. 1 的 右 图 所 
示 。 测 试 其 他 关键 词 将 取得 类 似 的 效果 。 
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关键 词 : 向 件 开始 搜索 


秩 多 是 一 家 美国 震 国 村 技 公 司 ， 伟 是 


和 PC(Persgonal Compunher ， 个 人 诗 入 
交 ) 软 件 到 发 的 和 导 ， 自 比尔 义演 与 傣 欧 
欠 便 办 于 1975 和 第， 公司 名 部 洽 立 在 轨 
= Ed C1 
如 保 广 的 志 脑 软件 骸 务 业务 为 地 

时 为 著名 和 负 詹 的 户 品 为 Mocrosoh 
ows 寻 人 作 夭 痢 扣 Mrcrosofl Offrxce 篆 
软件， 目前 是 全 未 是 大 的 电 议 软件 嫩 


PC(Personal Comenner ， 


芝 纶 创办 子 1975 年 ， 
以 杞 和 惧 血 ! 去 中 后 E 
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关键 词 : 的 开始 搜索 


粕 多 全 一 家 美国 振 国 村 技 公 司 ， 亿 是 
个 | 
和 狼 伟 交界 的 烙 导 ， 和 由 比 保 义 必 与 保 欧 
公司 多 部 注 立 在 和 
轿 州 的 震 实 莫 寝 ， 以 精 恬 、 制 半 、 类 权 
吓人 保 广 芭 的 电脑 竹 千林 基业 务 为 斑 

世 为 着 各 和 间 铺 铺 的 产品 为 Mscrosoft 
wdows 扶 作 系 统 阳 MCcrosofl Offrxce 系 


刚 久 任 ， 辣 前 是 全 未 量 大 的 电脑 匀 件 娃 侣 


图 069.1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

TextView myTextView; 

EditText myEditText; 

String myText; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
getWindow( ) . getDecorView( ) . setLayerType(View. LAYER TYPE SOFTWARE, null); 
setContentView(R. layout. activity main); 
myEditText = (EditText) findViewById(R. id. myEditText); 
myTextView = (TextView) findViewById(R. id.myTextView); 


myText = myTextView.getText().toString(); // 记 录 原 始 文 本 内 容 

} 

public void onClickButton(View v) { // 响 应 单 击 " 开 始 搜索 "按钮 
// 获 取 搜 索 字 符 串 (关键 词 ) 


String mySearch = myEditText.getText().toString(); 
SpannableString mySpannableString = new SpannableString(myText); 
for (int i = 0; i <myText.length(); i++) { 


int start = i; // 记 录 起 始 索 引 值 
i = myText. indexOf(mySearch，i); // 获 取 符 合 搜索 结果 的 索引 值 
if (i<0)f{ // 未 搜索 到 


mySpannableString. setSpan(new MaskFilterSpan(new BlurMaskFilter(8, 
BlurMaskFilter.Blur. NORMAL)), start + mySearch. length() - 1, myText. 
length(), Spannable. SPAN EXCLUSIVE EXCLUSIVE); 
break; 
} 
// 模 糊 显 示 不 符合 搜索 结果 的 部 分 , 以 默认 样式 显示 符合 搜索 结果 的 部 分 
mySpannableString. setSpan(new MaskFilterSpan(new BlurMaskFilter(8, 
BlurMaskFilter. Blur. NORMAL)), start + mySearch. length() - 1, 
i, Spannable. SPAN EXCLUSIVE EXCLUSIVE); 
} 
myTextView. setText(mySpannableString); // 应 用 SpannableString 对 象 
图， 


上 F 面 这 段 代 码 在 MyCode\ MySample799\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString. setSpan (new MaskFilterSpan (new 
BlurMaskFilter(8, BlurMaskFilter. Blur. NORMAL)), start 十 mySearch. length() - 1, myText. 
length() ，Spannable. SPAN_EXCLUSIVE_EXCLUSIVE) 表 示 对 指定 的 文本 进行 模糊 ,范围 从 
start 十 mySearch. length()-1 到 myText. length(), 模 糊 半 径 为 8 像素。 此 实例 的 完整 项 目 在 
MyCode\MySample799 文件 夹 中 ， 


070 使 用 BulletSpan 在 文本 首 字 前 沃 加 小 圆 品 


此 实例 主要 通过 使 用 BulletSpan ,实现 在 TextView 文本 的 首 字 前 添加 小 圆 点 符号 。 当 实例 运行 
之 后 ,在 TextView 文本 的 首 字 前 添加 小 圆 点 符号 的 效果 如 图 070. 1 所 示 。 
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MySample 


- 烃 酷 应 用 实例 集锦 


图 070. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
TextView myTextView = (TextView) findViewById(R. id. myTextView) ; 
SpannableString mySpannableString = 

new SpannableString(" 炫 酷 应 用 实例 集锦 " ); 

// 创 建 指定 颜色 的 符号 ,并 指定 符号 与 后 面 文字 之 间 宽 度 的 BulletSpan 


mySpannableString. setSpan(new BulletSpan( 50, 
Color. RED), 0, 1, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 


myTextView. setText(mySpannableString); 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample039\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString. setSpan(new BulletSpan (50 ,Color. 
RED), 0, 1, Spanned. SPAN_EXCLUSIVE EXCLUSIVE) 用 于 创建 一 个 距离 文本 首 字 50px 的 红 
色 小 圆 点 。 如 果 省 略 颜色 值 , 则 创建 的 小 圆 点 的 颜色 与 文本 的 颜色 相同 。 此 实例 的 完整 项 目 在 
MyCode\MySample039 文件 夹 中 ，。 


071 使 用 StrikethroughSpan 潜 加 文字 删除 线 


此 实例 主要 通过 使 用 StrikethroughSpan, 实 现在 TextView 中 的 部 分 文字 上 添加 删除 线 。 当 实 
例 运 行 之 后 便捷 的 ?三 个 字 的 中 间 有 一 条 删除 线 ,如 图 071. 1 所 示 。 
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C3 


MySample 


云 计算 是 一 种 按 使 用 量 付费 


的 模式 ， 这 种 模式 提供 可 用 
的 、 人 和 便捷 的 、 按 需 的 网 络 访 
问 ， 进 入 可 配置 的 计算 资源 
共享 闻 。 


图 071. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableString mySpannableString = new SpannableString(" 云 计算 是 一 种 按 使 用 量 付 费 的 模式 ,这 种 模式 
提供 可 用 的 便捷 的 、 按 需 的 网 络 访问 ,进入 可 配置 的 计算 资源 共享 池 ."); 
mySpannableString. setSpan(new StrikethroughSpan( ), 26, 29, 
Spanned. SPAN EXCLUSIVE EXCLUSIVE); // 在 "便捷 的 "上 设置 删除 线 
myTextView. setText(mySpannableString); 
}} 


上 面 这 段 代 码 在 MyCode\ MySample028\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString. setSpan(new StrikethroughSpan( ) ， 
26， 29，Spanned. SPAN_EXCLUSIVE EXCLUSIVE) 中 的 26 和 29 指示 删除 线 的 范围 , 即 “ 便 捷 的 ” 
三 个 字 在 字符 串 中 的 索引 位 置 。 如 果 需 要 为 “便捷 的 ?三 个 字 添 加 下 画 线 , 则 可 以 把 代码 修改 为 
myText. setSpan(new UnderlineSpan(), 26, 29, Spanned. SPAN_EXCLUSIVE _EXCLUSIVE ) 。 
此 实例 的 完整 项 目 在 MyCode\MySample028 文件 夹 中 ，。 


072 ”使 用 URLSpan 为 部 分 内 容 添 加 超 链 接 


此 实例 主要 通过 使 用 URLSpan ,实现 在 TextView 中 为 部 分 内 容 添 加 超 链接 。 当 实例 运行 之 后 ， 
“电话 “公司 网 站 ”和 “公司 地 址 ”将 以 超 链接 的 形式 显示 ,如 图 072. 1 的 左 图 所 示 ; 单 击 “电话 ” 超 链 


人 Androidg 本 可 a0og 。 实战 篇 


接 , 则 弹出 手机 的 电话 拨号 盘 ; 单 击 “ 公 司 网 站 ” 超 链接 , 则 打开 网 址 指定 的 网 站 ,如 果 手 机 上 有 多 个 浏 
览 融 , 则 显示 浏览 大 选择 窗口 ; 单 击 “ 公 司 地 址 ” 超 链接 , 则 通过 百度 地 图 (默认 地 图 工具 ) 定 位 指定 的 
地 址 ,如 图 072. 1 的 右 图 所 示 。 


MySample 


我 公司 以 国际 化 和 专业 化 为 
目标 ， 集 聚 了 一 批复 合 型 、 
开拓 型 的 营销 和 管理 等 各 方 
面 的 人 才 ， 你 可 以 通过 电话 
与 相关 人 员 进 行 联系 ， 也 可 
以 通过 公司 网 站 了 解 更 多 信 
息 ， 还 可 以 在 这 里 查看 公司 
地 址 。 


在 锦 天 星 都 附近 


精确 到 40 米 ,海拔 406 米 


他 收藏 急 搜 周边 回答 到 


图 072. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
Override 
protected void onCreate(Bundle savedInstanceState) { 

super. onCreate(savedInstanceState) ; 

setContentView(R. layout. activity_main); 

TextView myTextView = (TextView) findViewById(R. id.myTextView); 

SpannableString mySpannableString = new SpannableString( "我 公司 以 国际 化 和 专业 化 为 目标 ,集聚 了 一 批 
复合 型 .开拓 型 的 营销 和 管理 等 各 方面 的 人 才 , 你 可 以 通过 电话 与 相关 人 员 进 行 联系 , 也 可 以 通过 公司 网 站 了 解 
更 多 信息 ,还 可 以 在 这 里 查看 公司 地 址 . ") ; 

// 查 找 " 电 话 " 在 字符 串 中 的 索引 位 置 

int myStartPos = mySpannableString. toString(). indexof(" 电 话 ") ; 

int myEndPos = myStartPos + 2; 

mySpannableString. setSpan(new URLSpan("tel:13996060872" ), myStartPos, 

myEndPos, Spanned.SPAN EXCLUSIVE EXCLUSIVE); // 响 应 单 击 " 电 话 " 超 链 接 

// 查 找 "公司 网 站 "在 字符 串 中 的 索引 位 置 

myStartPos = mySpannableString. toString( ). index0f(" 公 司 网 站 "); 

myEndPos = myStartPos + 4; 

mySpannableString. setSpan(new URLSpan(" http://www. baidu. com" ), myStartPos, 

myEndPos, Spanned.SPAN EXCLUSIVE EXCLUSIVE); // 响 应 单 击 "公司 网 站 " 超 链接 

// 查 找 "公司 地 址 "在 字符 串 中 的 索引 位 置 

myStartPos = mySpannableString.toString(). indexOof(" 公 司 地 址 ") ; 

myEndPos = myStartPos + 4; 

// 响 应 单 击 "公司 地 址 " 超 链接 

mySpannableString. setSpan( new URLSpan(" geo:38.899533, - 77.036476" )，, 

myStartPos, myEndPos, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 


myTextView. setText(mySpannableString); 


// 实 现 单 击 超 链接 的 功能 
myTextView. setMovementMethod(LinkMovementMethod. getInstance( ) ); 
|} 


上 面 这 段 代 码 在 MyCode\ MySample033\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , mySpannableString. setSpan (new URLSpan ( "tel: 
13996060872"), myStartPos, myEndPos, Spanned. SPAN_EXCLUSIVE EXCLUSIVE) 用 于 设置 电 
话 超 链接 ,URLSpan("tel: 13996060872"”) 表 示 上 自动 根据 参数 内 容 调 用 手机 缺 省 应 用 ,如 果 该 参数 是 
“tel: 13996060872”, 则 调用 电话 拨号 盘 ; 如 果 该 参数 是 “http: // www. baidu. com” 网 址 格式 , 则 调 
用 浏览 器 ; 如 果 该 参数 是 “geo: 38. 899533 ,一 77. 036476” 地 址 格式 , 则 调用 地 图 工具 ; myStartPos 表 
示 超 链接 作用 范围 的 开始 位 置 , myEndPos 表示 超 链接 作用 范围 的 结束 位 置 。 但 是 仅 设 置 了 
URLSpan("tel: 13996060872"), 超 链接 不 会 起 作用 ,必须 调用 setMovementMethod(LinkMovement- 
Method. getInstance()) 方 法 , 超 链接 才 会 发 生 作 用 。 此 实例 的 完整 项 目 在 MyCode\MySample033 文 
件 夹 中 。 


073 ”使 用 ImageSpan 同时 显示 QQ 表情 和 文字 


此 实例 主要 通过 使 用 ImageSpan, 实 现在 TextView 中 混合 显示 QQ 表情 和 文字 。 当 实例 运行 之 
后 ,在 TextView 中 混合 显示 QQ 表情 和 文字 的 效果 如 图 073. 1 所 示 。 
QO 中 日 x 三 5 i 日 95% 下 午 5:50 


MySample 


> 


大 节 ， 伤 不 起 啊 ! 9》 伤 不 
起 啊 ! 


图 073.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(@ Override 
protected void onCreate( Bundle savedInstanceState) { 
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super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableString mySpannableString = 
new SpannableString(" 大 葡 , 伤 不 起 啊 ! 我 伤 不 起 啊 ! 伤 不 起 啊 !"); 
// 查 找 " 我 伤 不 起 啊 !" 在 字符 串 中 的 索引 位 置 
int myStartPos = mySpannableString.toString(). indexof(" 我 伤 不 起 啊 !"); 
int myEndPos = myStartPos + "我 伤 不 起 啊 !". toString(). length(); 
// 从 Drawable 获取 QQ 表情 资源 
Drawable myDrawable = getResources( ). getDrawable(R. drawable. myqq); 
myDrawable. setBounds(0, 0, myDrawable. getIntrinsicWidth( )/5, 
myDrawable. getIntrinsicHeight()/5); // 设 置 QQ 表情 图 像 的 大 小 
// 使 用 0 表情 图 像 蔡 换 " 我 伤 不 起 啊 !" 
mySpannableString. setSpan(new ImageSpan(myDrawable), 
myStartPos, myEndPos, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
myTextView. setText(mySpannableString); 
Fe 


上 面 这 上段 代码 在 MyCode\ MySample034\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 人 码 中 ,mySpannableString. setSpan (new ImageSpan 
(myDrawable), myStartPos, myEndPos, Spanned. SPAN EXCLUSIVE_EXCLUSIVE) 表 示 使 用 
QQ 表情 图 像 蔡 换文 字 “ 我 伤 不 起 啊 1”, myStartPos 和 myEndPos 表示 “我 伤 不 起 啊 1” 在 整个 字符 串 
中 的 起 止 位 置 。 此 实例 的 完整 项 目 在 MyCode\MySample034 文件 夹 中 。 


074 ”使 用 StyleSpan 实现 以 粗 斜体 显示 文字 


此 实例 主要 通过 使 用 StyleSpan, 实 现 以 粗 斜 体 风 格 显示 TextView 的 部 分 文字 。 当 实例 运行 之 
后 , “便捷 的 ”三 个 字 将 以 粗 斜 体 风格 显示 ,如 图 074. 1 所 示 。 


MySample 


云 计算 是 一 种 按 使 用 量 付费 


的 模式 ， 这 种 模式 提供 可 用 
的 、 伪 姜 的 、 按 需 的 网 络 访 
问 ， 进 入 可 配置 的 计算 资源 


共 侍 闻 。 


图 074. 1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableString mySpannableString = new SpannableString(" 云 计算 是 一 种 按 使 用 量 付费 的 模式 , 这 种 模式 
提供 可 用 的 、 便 捷 的 、 按 需 的 网 络 访 问 ,进入 可 配置 的 计算 资源 共享 池 ."); 
// 以 粗 和 斜体 风格 显示 "便捷 的 "三 个 字 
mySpannableString. setSpan( 
new StyleSpan(android. graphics. Typeface. BOLD ITALIC), 
26, 29, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
myTextView. setText(mySpannableString); 
三 


上 F 面 这 段 代 码 在 MyCode\ MySample029\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString. setSpan (new StyleSpan (android. 
graphics. Typeface. BOLD_ITALIC),26,29, Spanned. SPAN_ EXCLUSIVE_ EXCLUSIVE) 中 的 26 
和 29 指示 粗 和 斜体 的 作用 范围 ， 即 “便捷 的 ”三 个 字 在 字符 串 中 的 索引 位 置 ; android. graphics. 
Typeface. BOLD_ITALIC 表示 字体 是 粗 和 斜体 ; 如 果 此 参数 为 android. graphics. Typeface. ITALIC， 
则 指定 文字 仅 以 斜体 显示 ; 如 果 此 参数 为 android. graphics. Typeface. BOLD, 则 指定 文字 仅 以 粗 体 
显示 。 此 实例 的 完整 项 目 在 MyCode\MySample029 文件 夹 中 。 


075 ”使 用 SuperscriptSpan 绘制 勾 股 定理 公式 


此 实例 主要 通过 使 用 SuperscriptSpan 和 RelativeSizeSpan ,实现 在 TextView 中 以 标准 格式 显示 
勾 股 定理 公式 。 当 实例 运行 之 后 ,以 标准 格式 显示 勾 股 定理 的 效果 如 图 075. 1 所 示 。 


\ 站 


Am 


MySample 


勾 股 定理 公式 : a’+b”=c? 


图 075. 1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
TextView myTextView = (TextView) findViewById(R. id. myTextView) ; 
SpannableString mySpannableString = 
new SpannableString(" 勾 股 定理 公式 :a2 + b2 = c2"); 
// 以 上 标的 形式 显示 指数 
mySpannableString. setSpan(new SuperscriptSpan( ) ， 
8, 9, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
mySpannableString. setSpan(new SuperscriptSpan( ), 
11, 12, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
mySpannableString. setSpan(new SuperscriptSpan( ), 
14, 15, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
// 0.5f 表示 指数 字体 大 小 只 有 底数 字体 大 小 的 一 半 
mySpannableString. setSpan(new RelativeSizeSpan(0.5f), 
8, 9, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
mySpannableString. setSpan(new RelativeSizeSpan(0.5f), 
11, 12, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
mySpannableString. setSpan(new RelativeSizeSpan(0.5f), 
14, 15, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
myTextView. setText(mySpannableString); 
} } 


上 面 这 段 代 码 在 MyCode\ MySample030\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString. setSpan(new SuperscriptSpan() ,8， 
9， Spanned. SPAN_EXCLUSIVE_EXCLUSIVE) 用 于 将 指数 2 放 在 a 的 右上 和 角 , 此 时 指数 2 的 字体 
大 小 仍然 与 底数 a 相同 ; mySpannableString. setSpan(new RelativeSizeSpan(0. 5f) ,8，9 ，Spanned. 
SPAN_EXCLUSIVE _ EXCLUSIVE) 用 于 将 指数 2 的 字体 大 小 缩小 一 半 。 在 Android 中 ， 
SuperscriptSpan 用 于 以 上 标的 形式 重 置 文本 的 显示 位 置 , 它 一般 用 于 数学 公式 中 , RelativeSizeSpan 
用 于 根据 指定 的 参数 缩放 字体 大 小 。 此 实例 的 完整 项 目 在 MyCode\MySample030 文件 夹 中 。 


076 ”使 用 SubscriptSpan 绘制 硫酸 亚 铁 分 子 式 


此 实例 主要 通过 使 用 SubscriptSpan 和 RelativeSizeSpan ,实现 在 TextView 中 以 下 标 形式 显示 硫 
酸 亚 铁 的 分 子 式 。 当 实例 运行 之 后 ,以 下 标 形式 显示 的 硫酸 亚 铁 分 子 式 的 效果 如 图 076. 1 所 示 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableString mySpannableString = 

new SpannableString(" 硫酸 亚 铁 的 分 子 式 :FeS04" ); 

// 查 找 4 在 字符 串 中 的 索引 位 置 
int myFePos = mySpannableString.toString().indexOf("4"); 
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// 以 下 标的 形式 表示 FeS04 中 的 4 
mySpannableString. setSpan( new SubscriptSpan( ), 
myFePos, myFePos + 1, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
// 缩 小 下 标 字体 的 尺寸 
mySpannableString. setSpan(new RelativeSizeSpan(0.8f), 
myFePos, myFePos + 1, Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
myTextView. setText(mySpannableString); 
Ey 
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硫酸 亚 铁 的 分 子 式 : FeSO， 


图 076. 1 


上 F 面 这 段 代 码 在 MyCode\ MySample031 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 , mySpannableString. setSpan (new SubscriptSpan ( ) ， 
myFePos, myFePos 十 1, Spanned. SPAN EXCLUSIVE EXCLUSIVE) 用 于 将 4 以 下 标的 形式 放 在 
右 下 角 ,此 时 4 的 字体 大 小 不 变 。myFePos 和 myFePos 十 1 表示 SubscriptSpan() 作 用 范围 的 开始 
和 结束 索引 值 。mySpannableString. setSpan(new RelativeSizeSpan(0. 8f) ，myFePos，myFePos 十 
1 ,Spanned. SPAN_EXCLUSIVE_ EXCLUSIVE) 用 于 将 4 的 字体 大 小 缩小 至 原来 的 4/5。 在 
Android 中 ,SubscriptSpan 用 于 以 下 标的 形式 重 置 文本 的 显示 位 置 , 它 一 般 用 于 化 学 分 子 式 中 ， 
RelativeSizeSpan 用 于 根据 指定 的 参数 缩放 字体 大 小 。 此 实例 的 完整 项 目 在 MyCode\MySample031 
文件 夹 中 ，。 


077 ”使 用 TypefaceSpan 定制 文本 的 部 分 内 容 


此 实例 主要 通过 使 用 TypefaceSpan, 实 现 对 字符 串 中 的 部 分 文本 定制 个 性 化 的 字体 。 当 实例 运 
行 之 后 ,“ 微 软 风格 的 字体 好 看 呀 1” 中 的 “微软 ”二 字 将 以 自 定义 字体 显示 ,效果 如 图 077. 1 所 示 。 
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油 软 风格 的 字体 好 看 呀 ! 


图 077.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
myTextView. setText( "微软 风格 的 字体 好 看 呀 !"); 
Typeface myTypeface = Typeface. createFromAsset(getAssets(),"myfont.ttf"); // 加 载 指 定 字 体 
myTextView. setTypeface(myTypeface); // 在 TextView 控件 上 应 用 
该 字体 
SpannableString mySpannableString = new SpannableString(myTextView. getText() ); 
// 通 过 反 向 选取 区 域 并 应 用 字体 ,使 其 以 默认 字体 样式 显示 
mySpannableString. setSpan(new TypefaceSpan(" myfont" ), 2, 
myTextView. getText(). length(), Spanned. SPAN EXCLUSIVE EXCLUSIVE); 
myTextView. setText(mySpannableString); // 生 成 效果 并 显示 
门 ] 


上 F 面 这 段 代 码 在 MyCode\ MySample800\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySpannableString。 setSpan (new TypefaceSpan 
("myfont"), 2, myTextView. getText(). length()，Spanned. SPAN EXCLUSIVE EXCLUSIVE) 
表示 对 mySpannableString 字符 串 中 的 前 2 个 字符 使 用 字体 “myfont”。 需 要 说 明 的 是 , 自 定 义 字 体 文 
件 myfont. ttf 通常 放 在 MyCode\MySample800\app\src\main\ assets 目录 中 。 此 实例 的 完整 项 目 在 
MyCode\MySample800 文件 夹 中 ，。 


078 ”使 用 ForegroundColorSpan 创建 光照 文字 
此 实例 主要 通过 同时 使 用 ForegroundColorSpan 和 MaskFilterSpan, 使 字符 串 的 部 分 文本 呈现 


光照 的 浮雕 特效 。 当 实例 运行 之 后 ,同时 使 用 ForegroundColorSpan 和 MaskFilterSpan 产生 的 部 分 
文本 (Google) 浮 雕 特效 如 图 078. 1 所 示 。 


MySample 


Google 谷 歌 


图 078. 1 


public class MainActivity extends Activity { 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
myTextView. setText("Google 谷歌 ") ; 
SpannableString mySpannableString = new SpannableString(myTextView. getText() ); 
mySpannableString. setSpan(new ForegroundColorSpan( Color. RED), 
0, 6, Spanned. SPAN EXCLUSIVE EXCLUSIVE); // 应 用 红色 前 景 
mySpannableString. setSpan(new MaskFilterSpan( 
new EmbossMaskFilter(new float[ ]{1, 1, 1}, 0.25f, 6, 3.5f£)), 
0, 6, Spannable. SPAN EXCLUSIVE EXCLUSIVE); // 应 用 光照 特效 
myTextView. setText(mySpannableString); 
} } 


上 和 面 这 段 代码 在 MyCode\ MySample801 \app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,mySpannableString. setSpan(new ForegroundColorSpan 
(Color. RED) ,0, 6, Spanned. SPAN_EXCLUSIVE EXCLUSIVE) 表 示 对 mySpannableString 字符 
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串 的 前 6 个 字符 (“Google”) 应 用 红色 前 景 。mySpannableString. setSpan (new MaskFilterSpan(new 
EmbossMaskFilter(new float[ |{1, 1, 1}, 0.25f, 6, 3.5{)), 0, 6, Spannable. SPAN EXCLUSIVE 
_EXCLUSIVE) 表 示 对 mySpannableString 字符 串 的 前 6 个 字符 应 用 EmbossMaskFilter 产生 的 光照 
特效 。 在 测试 此 实例 时 ,通常 应 该 关闭 人 硬件 加 速 , 即 设置 android: hardwareAccelerated 一 "false" ,该 
代码 在 MyCode\MySample801\ app\src\main\AndroidManifest. xml 文件 中 。 此 实例 的 完整 项 目 在 
MyCode\MySample801 文件 夹 中 ，。 


079 ”使 用 BlurMaskFilter 创建 阴影 扩散 文字 


此 实例 主要 通过 在 自 定义 View 中 创建 BlurMaskFilter 模糊 滤 镜 ,并 在 画笔 Paint 中 使 用 
setMaskFilter() 方 法 设置 模糊 滤 镜 ,实现 绘制 阴影 扩散 的 文字 。 当 实例 运行 之 后 ,在 自 定义 View 中 
使 用 BlurMaskFilter 模糊 滤 镜 绘制 阴影 扩散 的 文字 的 效果 如 图 079. 1 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
(@Override MySample 
protected void onCreate( Bundle savedInstanceState) { 
setContentView(new MyView(this)); 
super. onCreate( savedInstanceState) ; 
} 
class MyView extends View { 
private Paint myPaint; 
public MYView(Context context) { 
super (context ) ; 
myPaint = new Paint( ); 
myPaint. setFlags(Paint. ANTI ALIAS FLAG); 
myPaint. setAntiAlias(true); 
myPaint. setColor(Color. BLACK); 
myPaint. setTextSize(220); 
myPaint. setStyle(Paint. Style. FILL AND STROKE); 
myPaint. setStrokeWidth(2); 
BlurMaskFilter myBlurMaskFilter = 
new BlurMaskFilter(40, BlurMaskFilter. Blur. SOLID); 
// 创 建 模 糊 滤 镜 
myPaint. setMaskFilter(myBlurMaskFilter);  // 设 置 模 糊 滤 镜 画 笔 图 079.1 
} 
(Override 
protected void onDraw(Canvas myCanvas) { 
super. onDraw( myCanvas ) ; 
Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay.getWidth( ); 
int myHeight = myDisplay. getHeight( ); 
myCanvas. drawText(" 炫 酷 实例 " ，myWidth/10， 
myHeight/2 - 150，myPaint); // 显 示 扩 散 阴 影 文本 


全 二 


上 F 面 这 段 代 码 在 MyCode\ MySample087\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myBlurMaskFilter = new BlurMaskFilter (40， 
BlurMaskFilter. Blur. SOLID) 用 于 创建 模糊 滤 镜 ,40 表示 模糊 半径 ,BlurMaskFilter. Blur，SOLID 表 
示 模 糊 模 式 是 SOLID。BlurMaskFilter 支持 下 列 4 种 模糊 模式 。 
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(1) NORMAL ,表示 同时 绘制 图 形 本 号 内 容 十 内 阴影 十 外 阴影 ,正常 阴影 效果 。 
(2) INNER ,表示 绘制 图 形 内 容 本 身 十 内 阴影 ,不 绘制 外 阴影 。 

(3) OUTER ,表示 不 绘制 图 形 内 容 以 及 内 阴影 ,只 绘制 外 阴影 。 

(4) SOLID ,表示 只 绘制 外 阴影 和 图 形 内 容 本 身 , 不 绘制 内 阴影 

此 实例 的 完整 项 目 在 MyCode\MySample087 文件 夹 中 ， 


080 ”使 用 EmbossMaskFilter 创建 浮雕 文字 


此 实例 主要 通过 使 用 浮雕 滤 镜 EmbossMaskFilter, 实 现在 TextView 中 创建 浮雕 风格 的 文本 。 
当 实 例 运 行 之 后 ,在 TextView 中 显示 的 浮雕 文本 的 效果 如 图 080. 1 所 示 。 


MySample 


ALE 


图 080.1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
SpannableStringBuilder mySpannableStringBuilder = 


new SpannableStringBuilder(" 炫 酷 " ); 
float[] myDirection = new float[]{ 10, 10, 10 }; // 设 置 光源 的 方向 
float myLight = 0.1f; // 环 境 光 亮度 
float mySpecular = 5; // 反 射 等 级 
float myBlur = 5; // 模 类 半径 


// 根 据 指定 的 参数 创建 浮雕 滤 镜 EmbossMaskFilter 
EmbossMaskFilter myEmbossMaskFilter = 
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new EmbossMaskFilter(myDirection, myLight, mySpecular, myBlur); 


// 根 据 浮 雕 滤 镜 创建 MaskFilterSpan 
MaskFilterSpan myMaskFilterSpan = new MaskFilterSpan(myEmbossMaskFilter); 
mySpannableStringBuilder. setSpan(myMaskFilterSpan,0, 2, 
Spanned. SPAN_INCLUSIVE INCLUSIVE); // 设 置 MaskFilterSpan 的 作用 范围 
myTextView. setText(mySpannableStringBuilder); 
} } 


上 面 这 上段 代码 在 MyCode\ MySample038\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myEmbossMaskFilter 二 new EmbossMaskFilter 
(myDirection, myLight, mySpecular, myBlur) 用 于 根据 指定 的 参数 myDirection，myLight， 
mySpecular，myBlur 创建 浮雕 滤 镜 ; 当 浮 雕 滤 镜 创建 后 ,再 使 用 该 浮雕 滤 镜 创建 MaskFilterSpan, 即 
myMaskFilterSpan 一 new MaskFilterSpan (myEmbossMaskFilter ) 。 mySpannableStringBuilder. 
setSpan(myMaskFilterSpan, 0, 2, Spanned. SPAN INCLUSIVE_ INCLUSIVE) 中 的 0 表示 浮雕 滤 
镜 发 生 作 用 的 开始 位 置 索引 ,2 表示 浮雕 滤 镜 发 生 作 用 的 结束 位 置 索引 。 在 测试 此 实例 时 , 通 第 应 该 
关闭 硬件 加 速 , 即 在 MyCode\ MySample038\app\src \main\AndroidManifest. xml 文件 中 设置 
android: hardwareAccelerated= 王 "false" 。 此 实例 的 完整 项 目 在 MyCode\MySample038 文件 夹 中 ， 


081 ”通过 自 定 义 View 在 半圆 疲 上 绘制 文字 
此 实例 主要 通过 使 用 Path 的 addArc() 方 法 和 Canvas 的 drawTextOnPath() 方 法 ,在 自 定义 


View 中 实现 在 半圆 弧 路 径 上 绘制 文字 。 当 实例 运行 之 后 ,在 半圆 弧 路 径 上 绘制 的 文字 “Android 炫 酷 
应 用 实例 集锦 ”的 效果 如 图 081. 1 所 示 。 


MySample 


图 081.1 


主要 代码 如 下 : 


< com. bin. luo. mysample. CustomTextView android:layout width= "match parent” 
android: layout height = "wrap_content"/> 


上 面 这 段 代 码 在 MyCode\MySample775\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,com. bin. luo. mysample. CustomTextView 是 自 定义 View 类 (控件 ) 在 布局 文件 中 的 
应 用 ,com. bin. luo. mysample 是 包 名 ,在 实际 应 用 中 通常 应 该 修改 为 自己 的 包 和 名 , 自 定义 类 
CustomTextView 的 主要 代码 如 下 : 


public class CustomTextView extends View { 
String myText = "Android 炫 酷 应 用 实例 集锦 "; 
Paint myPaint null; 
Path myPath = null; 
public CustomTextView(Context context, AttributeSet attrs) { 
super(context, attrs); 
ET 
} 
public void Init() { // 初 始 化 画笔 和 路 径 对 象 
myPaint = new Paint(); 
myPaint. setAntiAlias(true); 
myPaint. setTextSize( 96); 
LinearGradient myGradient = new LinearGradient(100,100, 800, 800, 
Color. GREEN，Color. BLUE，Shader.TileMode.MIRROR);  // 初 始 化 线性 渐变 对 象 


myPaint. setShader(myGradient ); // 使 用 渐变 色 填 充 画笔 颜色 
if (myPath == null) { // 若 未 设置 路 径 对 象 , 则 默认 将 以 半圆 路 径 显 
示 文 本 
myPath = new Path(); 
myPath. addArc(100,100, 800, 800,180, 180); // 绘 制 半 圆 路 径 
} } 
(@Override 


protected void onDraw(Canvas canvas) { 
super. onDraw(canvas ) ; 


canvas. translate( 100, 490); // 平 移 画 布 至 指定 位 置 
canvas. drawTextOnPath(myText, myPath, 3, 
-10, myPaint); // 在 指定 路 径 上 绘制 指定 文本 
} } 


上 面 这 段 代 码 在 MyCode\ MySample775\app\src\ main\java\com\bin\luo\mysample\ 
CustomTextView. java 文件 中 。 在 这 段 代 码 中 ,myGradient 一 new LinearGradient(100,100，800， 
800，Color. GREEN , Color. BLUE ,， Shader. TileMode. MIRROR) 用 于 创建 一 个 填充 Paint 的 线性 渐 
变色 对 象 ,LinearGradient() 构 造 曙 数 的 语法 声明 如 下 : 


public LinearGradient(float x0, float y0, float xl， 
float yl, int color0, int colorl, TileMode tile) 


其 中 ,参数 float x0 表示 渐变 起 点 的 xz 坐标; 参数 float y0 表示 渐变 起 点 的 y 坐标 ; 参数 float xl 
表示 渐变 终点 的 工 坐标 ; 参数 float yl 表示 渐变 终点 的 y 坐标 ; 参数 int color0 表示 渐变 开始 颜色 ; 
参数 int colorl 表示 渐变 结束 颜色 ; 参数 TileMode tile 表示 平 铺 方式 。TileMode 有 3 种 参数 可 供 选 
择 , 分 别 为 CLAMP、REPEAT 和 MIRROR; CLAMP 的 作用 是 如 果 泻 染 带 超出 原始 边界 范围 , 则 会 
复制 边缘 颜色 对 超出 范围 的 区 域 进行 着 色 ,REPEAT 的 作用 是 在 横向 和 纵向 上 以 平 铺 的 形式 重复 泻 
染 ,MIRROR 的 作用 是 在 横向 和 纵 癌 上 以 镜像 的 方式 重复 演 染 。 
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myPath. addArc(100,100, 800, 800,180, 180) 的 作用 是 绘制 一 个 半圆 弧 路 径 。addArc0O) 方 法 的 
语法 声明 如 下 : 


public void addArc(float left, float top, 
float right, float bottom, float startAngle, float sweepAngle) 


其 中 ,参数 float left 表示 圆 弧 外 接 和 矩形 左上 角 的 工 坐标 ,参数 float top 表示 圆 弧 外 接 和 矩形 左上 角 
的 > 坐标 ,参数 float right 表示 圆 弧 外 接 和 矩形 右 下 角 的 xz 坐标 ,参数 float bottom 表示 圆 弧 外 接 和 矩形 
右 下 角 的 y 坐标 ,float startAngle 表示 圆 弧 起 始 角 ,此 角度 不 好 理解 , 它 以 水 平 线 右 端 为 0 ,以 顺 时 针 
方 回 为 增 量 ,此 实例 所 用 的 180" 即 在 水 平 线 的 左 端 ,float sweepAngle 表示 圆 弧 的 度数 范围 ,在 此 实例 
中 ,startAngle 为 180, sweepAngle 为 180, 因 此 终止 角 为 360", 即 水 平 线 的 右 端 ”如 果 myPath. 
addArc(9, 9，350，350,90,180), 则 圆 弧 文字 表现 为 左 端 ,而 不 是 上 端 。 

canvas. drawTextOnPath(myText, myPath, 3, 一 10, myPaint) 表 示 在 指定 的 圆 弧 路 径 上 绘制 
渐变 色 文 字 。drawTextOnPath() 方 法 的 语法 声明 如 下 : 


public void drawTextOnPath((@NonNul11l String text, (ONonNull Path path, 
float hOffset，float vOffset, (WNonNull Paint paint) 


其 中 ,参数 String text 表示 显示 的 文字 ,参数 Path path 表示 文字 的 显示 路 径 , 参 数 float hOffset 
表示 水 平 偏 移 量 ,参数 float vOffset 表示 垂直 偏 移 量 ,参数 Paint paint 表示 定制 的 画笔 。 
此 实例 的 完整 项 目 在 MyCode\MySample775 文件 夹 中 。 


082 ”通过 自 定 义 View 在 圆 约 上 深 动 文字 


此 实例 主要 通过 使 用 Timer 定时 修改 drawTextOnPath() 方 法 的 hOffset 参数 (水 平 偏 移 量 ), 实 
现 使 文字 沿 着 自 定 义 圆 弧 路 径 滚动 显示 。 当 实例 运行 之 后 , 单 击 “ 开 始 滚动 ?按钮 , 则 演示 文本 ”人生 
得 意 须 尽 欢 ? 将 沿 着 自 定 义 圆 弧 路 径 以 顺 时 针 方 回 滚 动 显示 ; 单 击 "暂停 滚动 按钮, 则 滚动 显示 停止 ; 
再 次 单 击 “ 开 始 滚动 ?按钮 , 则 演示 文本 将 从 上 次 暂停 的 位 置 继续 滚动 显示 ,效果 分 别 如 图 082. 1 的 左 
图 和 右 图 所 示 。 
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图 082.1 


主要 代码 如 下 : 


// 响 应 单 击 "开始 滚动 "按钮 

public void onClickBtn]l (View v){ myCustomTextView. startMarqueeAnim( ); } 
// 响 应 单 击 "暂停 滚动 "按钮 

public void onClickBtn2(View v){ myCustomTextView. pauseMaruqeeAnim( ) ; } 


上面 这 段 代 码 在 MyCode\ MySample776\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ， myCustomTextView 三 (CustomTextView ) 
findViewBylId(R. id. myCustomTextView) 中 的 CustomTextView 是 自 定义 控件 (类 ), 该 控件 (类 ) 的 
主要 代码 如 下 : 


public class CustomTextView extends View { 

String myText = "人 生得 意 须 尽 欢 "，; 

Paint myPaint null; 

Path myPath = null; 

Timer myTimer; 

TimerTask myTask; 

float myPositionX = 0; 

public CustomTextView(Context context, AttributeSet attrs) { 
super(context, attrs); 
nlitetys 

} 

public void Init() { // 初 始 化 画笔 和 路 径 对 象 
myPaint = new Paint(); 
myPaint. setAntiAlias(true); 
myPaint. setTextSize( 80); 
if (myPath == null) { // 若 未 设置 路 径 对 象 , 则 默认 将 以 半圆 路 径 显 示 文 本 

myPath = new Path(); 
myPath. addArc(10, 10, 700, 700, 0, 360); “// 绘 制 半圆 路 径 

}} 

(Override 

protected void onDraw(Canvas canvas) { 
super. onDraw(canvas ) ; 


canvas. translate(150, 300); // 平 移 画 布 至 指定 位 置 
canvas. drawTextOnPath(myText, myPath, 

myPositionX, - 10, myPaint); // 在 指定 路 径 上 绘制 指定 文本 
} 
public void startMarqueeAnim() { // 创 建 定时 任务 


if (myTask != null) { myTask. cancel(); } 


myTimer = new Timer( ); 


myTask = new CustomTimerTask(); // 初 始 化 自 定义 定时 任务 类 
myTimer. schedule(myTask, 0, 100); // 提 交 任 务 至 定时 器 并 开始 执行 
} 

// 取 消 定时 任务 


public void pauseMarugqeeAnim() { myTask. cancel( ); } 
private class CustomTimerTask extends TimerTask { 
(QOverride 
public void run() { 
myPositionX += 10; 
if (myPositionX == 1500) { myPositionX = 0; } 
postInvalidate!( ); 
| 


上 面 这 段 代 码 在 MyCode\ MySample776\app\src\ main\java\com\bin\luo\mysample\ 
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CustomTextView. java 文件 中 。 在 这 段 代 码 中 ,myTimer. schedule(myTask, 0，100) 表 示 间 隔 执 行 
滚动 显示 文本 任务 ,schedule() 方 法 的 语法 声明 如 下 : 


public void schedule(TimerTask task, long delay, long period) 


其 中 ,参数 TimerTask task 是 一 个 TimerTask 派生 类 的 实例 ,该 派生 类 通常 需要 实现 public 
void run() 方 法 ,因为 TimerTask 类 实现 了 Runnable 接口 。 人 参数 long delay 用 于 设置 timer 定时 髓 
第 一 次 调用 run 方法 的 时 间 ,0 表示 不 设置 时 间 ,立刻 执行 任务 。 参 数 long period 表示 第 一 次 执行 任 
务 之 后 ,从 第 二 次 开始 每 隅 多 长 的 时 间 调 用 一 次 run() 方 法 ,以 ms 为 单位 。 

此 实例 的 完整 项 目 在 MyCode\MySample776 文件 夹 中 。 


083 ”通过 自 定 义 View 绘制 渐变 色 的 文字 


此 实例 主要 通过 使 用 线性 渐变 LinearGradient 创建 着 色 器 Shader, 并 在 画笔 Paint 中 使 用 
setShader() 方 法 设置 Shader ,实现 绘制 颜色 渐变 的 文字 。 当 实例 运行 之 后 ,在 自 定 义 View 中 绘制 的 
颜色 渐变 文字 的 效果 如 图 083. 1 所 示 。 


4 日 上 上午 8:16 
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图 083. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
setContentView(new MYView(this) ); 
super. onCreate( savedInstanceState) ; 


} 
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class MyView extends View { 

public MyView(Context context) { super(context); } 

(QOverride 

protected void onDraw(Canvas myCanvas) { 
Display myDisplay = getWindowManager( ) . getDefaultDisplay( ); 
int myWidth = myDisplay.getWidth( ); // 获 取 屏 幕 宽 度 
int myHeight = myDisplay.getHeight(); // 获 取 屏 幕 高 度 
Shader myShader = new LinearGradient(0, 0, 160, 160, 

new int[ ] {Color. RED, Color.GREEN, Color.BLUE, Color. YELLOW}, 


null, Shader.TileMode. REPEAT); // 创 建 渐变 色 的 着 色 器 
Paint myPaint = new Paint( ); 
myPaint. setShader (myShader); // 使 用 渐变 色 着 色 器 设置 画笔 
myPaint. setTextSize(160); // 设 置 文字 大 小 
myCanvas. drawColor (Color. WHITE); // 设 置 背 景 颜色 


myCanvas. drawText(" 炫 酷 应 用 实例 ",，50， 
myHeight / 2 - 200, myPaint); // 绘 制 渐变 色 文 字 
} 
} 
} 


上 和 面 这 上段 代码 在 MyCode\ MySample085\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myShader 二 new LinearGradient(0, 0, 160, 160, new 
int[L ] {Color. RED，Color. GREEN ，Color. BLUE, Color. YELLOW)}, null, Shader. TileMode. 
REPEAT) 用 于 创建 渐变 色 的 着 色 需 。myCanvas. drawText(" 炫 酷 应 用 实例 ", 50, myHeight / 2 一 
200 ,myPaint) 中 的 50 表示 绘制 文字 开始 位 置 的 水 平 坐标 ,myHeight / 2 一 200 表示 绘制 文字 开始 
位 置 的 垂直 坐标 。 此 实例 的 完整 项 目 在 MyCode\MySample085 文件 夹 中 。 


084 ”通过 自 定义 View 绘制 线条 描 边 文字 


此 实例 主要 通过 设置 画笔 的 样式 为 STROKE ,实现 在 自 定 义 View 中 绘制 线条 描 边 的 空心 文字 。 
当 实 例 运 行 之 后 ,在 自 定 义 View 中 绘制 的 线条 描 边 的 空心 文字 的 效果 如 图 084. 1 所 示 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override MySample 
Protected void onCreate(Bundle savedInstanceState) { 
setContentView(new MYView(this) ); 
super. onCreate( savedInstanceState) ; 
} 
class MyView extends View { 
private Paint myPaint; 
public MyView(Context context) { 


super(context ) ; 

myPaint = new Paint(Paint.ANTI ALIAS FLAG); 

myPaint. setColor(Color. BLUE); // 设 置 文字 颜色 
myPaint. setStyle(Paint. Style. STROKE); // 创 建 空心 文字 
myPaint. setStrokeWidth(1); // 设 置 文字 线条 宽度 
myPaint. setTextSize(220); // 设 置 字体 大 小 

} 

(QOverride 


protected void onDraw(Canvas myCanvas) { 
super. onDraw( myCanvas ) ; 
Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay. getWidth(); 084.1 
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int myHeight = myDisplay. getHeight( ); 
myCanvas. scale(1.0f,1.5f); // 在 垂直 方向 拉 伸 1.5 售 
myCanvas. drawText(" 炫 酷 实 例 ", myWidth/10, myHeight * 3/10, myPaint); 
} } 
} 


上 面 这 上段 代码 在 MyCode\ MySample088\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPaint. setStyle(Paint. Style. STROKE ) 用 于 设置 画笔 
的 样式 为 空心 。 在 Android 中 ,Paint. Style 用 于 设置 画笔 的 样式 , 取 值 如 下 。 

(1) FILL, 该 样式 用 于 创建 实心 画笔 。 

(2) FILL_AND_STROKE ,该 样式 用 于 使 画笔 同时 实现 实心 和 空心 效果 ,该 样式 在 某 些 场合 会 
市 来 不 可 预期 的 显示 效果 。 

(3) STROKE ,该 样式 用 于 创建 空心 画笔 。 

此 实例 的 完整 项 目 在 MyCode\MySample088 文件 夹 中 。 


085 ”通过 自 定 义 View 绘制 阴影 扩散 文字 


此 实例 主要 通过 使 用 setShadowLayer() 方 法 ,实现 在 自 定 义 View 中 绘制 阴影 扩散 的 文字 。 当 
实例 运行 之 后 ,在 自 定义 View 中 绘制 的 阴影 扩散 文字 的 效果 如 图 085. 1 所 示 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
@Override se 
protected void onCreate( Bundle savedInstanceState) { 
setContentView(new MyView(this)); 
super. onCreatel( savedInstanceState); 
} 
class MyView extends View { 
private Paint myPaint; 炫 酵 实例 
public MyView(Context context) { 
super(context); 
myPaint = new Paint( ); 
myPaint. setFlags(Paint. ANTI ALIAS FLAG); 
myPaint. setAntiAlias(true); 
myPaint. setColor(Color. BLUE); 
myPaint. setTextSize(200); 
myPaint. setStyle(Paint. Style. FILL AND STROKE); 
myPaint. setStrokeWidth(2); 
myPaint. setShadowLayer(15f, 10f, 10f, Color.GRAY); // 设 置 阴影 画笔 
} 
(QOverride 
protected void onDraw( Canvas myCanvas) { 
Super. onDraw(myCanvas ) ; 
Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay. getWidth( ); 
int myHeight = myDisplay. getHeight( ); 
myCanvas. drawText(" 炫 酷 实 例 ", myWidth / 10, 
myHeight x 2 / 5, myPaint); // 绘 制 阴影 文字 


Pe 


上 F 面 这 段 代 码 在 MyCode\ MySample091\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPaint. setShadowLayer(15f,10f,10f，Color. GRAY ) 用 
于 设置 阴影 画笔 ,15{ 表示 扩散 半径 ,10f 表示 水 平方 向 的 偶 移 量 ,10{ 表示 垂直 方向 的 偏 移 量 ，Color. 
GRAY 表示 阴影 颜色 。 此 实例 的 完整 项 目 在 MyCode\MySample091 文件 夹 中 。 


086 ”加载 字 库 文 件 显示 自 定 义 草 书 字体 


此 实例 主要 通过 使 用 Typeface 的 createFromAsset() 方 法 根据 指定 的 草 体 字库 文件 创建 自 定 义 
字体 ,实现 以 草 体 字 显示 文本 。 当 实例 运行 之 后 , 单 击 “ 以 默认 字体 显示 ”按钮 , 则 唐诗 将 以 默认 的 黑 
体 字 显示 ,如 图 086. 1 的 左 图 所 示 。 单 击 “ 以 草 体 显示 ”按钮 , 则 唐诗 将 以 自 定义 的 草 体 字 显 示 , 如 
图 086. 1 的 右 图 所 示 。 
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以 默认 字体 显示 以 草 体 显示 以 默认 字体 显示 以 草 体 显示 


全 外 忽 传 收 便 北 ， 
人 急 闻 闭 站 满 衣 受 。 
却 看 妻子 么 何在 ， 
瘟 和 益 诗 书 喜 欲 狂 。 


日 日 放歌 须 纵 酒 ， 
青春 作 伴 好 还 乡 。 
即 从 巴 峡 罕 俱 峡 ， 
便 下 到 阳 同 阁 阳 。 


图 086. 1 
主要 代码 如 下 : 


// 响 应 单 击 "以 默认 字体 显示 "按钮 
public void onClickmyBtnl (View v){ myTextView. setTYpeface(TYpeface. DEFRAULT) ;} 
// 响 应 单 击 "以 草 体 显示 "按钮 
public void onClickmyBtn2(View v) { 
Typeface myFont = Typeface. createFromAsset(this.getAssets(), "myfont.ttf" ); 
myTextView. setTypeface( myFont ); // 加 载 自 定义 草 体 字 
} 


上 面 这 段 代码 在 MyCode\ MySample644\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myFont 二 Typeface. createFromAsset (this. getAssets()， 
"myfont. ttf") 用 于 根据 草 体 字库 文件 myfont. ttf 创建 草 体 字 。 需 要 说 明 的 是 ,如 果 当 前 项 目 不 存 在 
assets 子 目 录 , 应 该 首先 在 app\src\main 目录 下 创建 assets 子 目录 ,然后 在 assets 子 目 录 中 添加 草 体 
字库 文件 myfont. ttf。 此 实例 的 完整 项 目 在 MyCode\MySample644 文件 夹 中 。 


CSy Androidg 本 可 a0og 。 实战 篇 


087 ”加 载 字 库 文 件 显 示 自 定义 液晶 字体 


此 实例 主要 通过 在 assets 目录 中 添加 自 定 义 液晶 字体 ,并 使 用 createFromAsset() 方 法 获取 此 字 
体 ,实现 在 TextView 中 显示 自 定义 的 液晶 字体 文字 。 当 实例 运行 之 后 ,“HTML AND CSS” 以 液晶 
字体 显示 的 效果 如 图 087. 1 所 示 。 


图 087.1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 


AssetManager myAssetManager = getAssets(); // 获 取 AssetManager 
Typeface myTypeface = Typeface.createFromAsset(myAssetManager, 
"myLed. TTF" ); // 根 据 液晶 字体 路 径 获 取 Typeface 
myTextView. setTypeface(myTypeface); // 设 置 当前 字体 为 液晶 字体 
三 


上 F 面 这 段 代 码 在 MyCode\ MySample019\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,setTypeface(myTypeface) 用 于 设置 myTypeface 字体 为 
当前 文字 的 字体 ,myTypeface 一 Typeface. createFromAsset(myAssetManager, "myLed. TTF") 则 
用 于 将 myLed. TTF 字体 文件 以 资源 的 形式 导入 字体 库 。 此 实例 的 完整 项 目 在 MyCode\\ 
MySample019 文件 夹 中 。 


088 ”判断 在 一 个 字符 串 中 是 否 包含 汉字 


此 实例 主要 通过 使 用 Pattern 和 Matcher 的 成 员 方 法 ,实现 根据 正则 表达 式 “[\u4E00 一 \u9FA5\ 
uF900 一 \uFA2Dj” 判 断 在 一 个 字符 串 中 是 否 包 含 汉 字 。 当 实例 运行 之 后 ,如 果 在 “测试 内 容 : “输入 
框 中 输入 “China 是 一 个 伟大 的 国家 ”, 然 后 单 击 “ 检 测 该 字符 串 是 否 包 含 汉 字 ” 按 钮 , 则 在 弹出 的 
Toast 中 提示 “该 字符 串 有 汉字 !1”, 如 图 088. 1 的 左 图 所 示 。 如 果 在 “测试 内 容 : ”输入 框 中 输入 
“China is a great country”, 然 后 单 击 “检测 该 字符 串 是 否 包 含 汉 字 ? 按 钮 , 则 在 弹出 的 Toast 中 提示 
“该 字符 串 没有 汉字 !1”, 如 图 088. 1 的 右 图 所 示 。 
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| 试 内 容 ;: China 是 一 个 伟大 的 国家 | 试 内 容 : China is a great countny 


检测 该 字符 串 是 否 包含 汉字 检测 该 字符 串 是 否 包含 汉字 


主要 代码 如 下 : 


public void onClickmyBtnl(View v) { // 响 应 单 击 "检测 该 字符 串 是 否 包含 汉字 "按钮 
String myText = myEditText.getText().toString(); 
final String format = "[\\u4E00 - \\u9FAS\\uF900 - \\uFA2D]"; 
Pattern myPattern = Pattern.compile(format); 


Matcher myMatcher = myPattern.matcher(myText); 
boolean myResult = myMatcher. find( ); 
if(myResult){ 


Toast. makeText (getApplicationContext( ), 


"该 字符 串 有 汉字 !" ，Toast. LENGTH SHORT). show(); 
}else{ 


Toast. makeText (getApplicationContext(), 
"该 字符 串 没 有 汉字 !"，Toast. LENGTH_SHORT). show( ); 
站 


上 面 这 段 代 码 在 MyCode\ MySample539\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myMatcher. find() 的 返回 值 如 果 为 true, 表 示 测 试 内容 与 
正则 表达 式 匹 配 ; 如 果 为 false, 表 示 测 试 内 容 与 正则 表达 式 不 匹配 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample539 文件 夹 中 。 


图 形 和 图 像 


089 在 自 定 义 View 中 绘制 径 回 渐变 的 图 形 


此 实例 主要 通过 使 用 RadialGradient 类 ,实现 在 自 定 义 View 中 绘制 径 回 渐变 的 图 形 。 当 实例 运 
行 之 后 ,在 自 定义 View 中 绘制 的 径 回 渐变 图 形 如 图 089. 1 所 示 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { 回电 日 时 
Override yng 
protected void onCreate(Bundle savedInstanceState) { 

setContentView(new MyView(this)); 
super. onCreate( savedInstanceState) ; 
} 
class MyView extends View { 
public MyView(Context context) { super(context); } 
人 Override 
protected void onDraw(Canvas myCanvas) { 
Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay. getWidth( ); 
int myHeight = myDisplay. getHeight( ); 
Paint myPaint = new Paint(); 
myPaint. setFlags(Paint. ANTI ALIAS FLAG); 
myPaint. setAntiAlias(true); 图 089.1 
RadialGradient myGradient = new RadialGradient (myWidth / 2, 
myHeight / 2 - 160, myWidth / 2 - 20, new int[ ]{Color.YELLOW, Color.GREEN, 
Color. TRANSPARENT, Color.RED}, null, Shader.TileMode. REPEAT); 
myPaint. setShader (myGradient ); // 设 置 径 向 渐变 画笔 
myCanvas. drawCircle(myWidth / 2, myHeight / 2 - 160, 
myWidth / 2 - 20, myPaint); // 绘 制 径 向 渐变 的 圆 形 


Ti 


上 面 这 上段 代码 在 MyCode\ MySample092\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myGradient 一 new RadialGradient (myWidth / 2， 
myHeight / 2 一 160, myWidth / 2 一 20, new int[ |{Color. YELLOW, Color. GREEN , Color. 
TRANSPARENT, Color. RED)}，null，Shader. TileMode. REPEAT) 用 于 创建 一 个 径 向 渐变 
RadialGradient, 该 构造 阴 数 的 语法 声明 如 下 : 


Sf 
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public RadialGradient(float x, float y, float radius, 
int[ ] colors, float[ ] positions, Shader. TileMode tile) 


其 中 ,参数 float x 表示 圆心 zx 坐标 ,参数 float y 表示 圆心 y 坐标 ,参数 float radius 表示 径 回 渐变 
的 半径 ,参数 int[ ] colors 表示 泻 染 颜色 数组 ,参数 float[ ] positions 表示 相对 位 置 数组 ,可 为 null, 即 
颜色 沿 渐变 方向 均匀 分 布 ,参数 Shader. TileMode tile 表示 这 染 平 铺 模式 ,包括 CLAMP MIRROR 
和 REPEAT 3 种 模式 。 


此 实例 的 完整 项 目 在 MyCode\MySample092 文件 夹 中 ， 


090 ”在 自 定义 View 中 实现 图 像 波 纹 起 伏 效 果 


此 实例 主要 通过 使 用 drawBitmapMesh() 方 法 绘制 扭曲 图 像 ,实现 图 像 的 波纹 起 伏 效果 。 当 实例 
运行 之 后 ,图像 将 会 像 波 浪 一 样 上 下 起 伏 ,效果 分 别 如 图 090. 1 的 左 图 和 右 图 所 示 。 


图 090. 1 


主要 代码 如 下 : 


<LinearLayout android:]layout width= "match parent" 
android:layout height = "match parent" 
android:orientation = "vertical"> 
<com.bin. luo.mysample.myImageView android:layout width = "wrap content" 
android:layout height = "wrap_content" 
android: scaleType = "center" 
android:visibility = "visible"/> 
</LinearLayout > 


上 面 这 段 代码 在 MyCode\MySamplel46\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 人 码 中 ,com. bin. luo. mysample. myImageView 即 是 用 于 实现 图 像 波 浪 起 伏 的 自 定义 控件 ， 
myImageView 类 (控件 ) 的 主要 代码 如 下 : 
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public class myImageView extends View { 
// 定 义 两 个 常量 表示 需要 将 图 像 划 分 成 多 少 个 小 方 格 


private int meshWidth = 40; // 横 向 划分 的 方 格 数目 
private int meshHeight = 40; // 纵 向 划分 的 方 格 数目 
private float FREQUENCY = 0.1f; // 三 角 函 数 的 频率 大 小 
private int AMPLITUDE = 80; // 三 角 函 数 的 振幅 大 小 


private int myCount = (meshWidth + 1) x* (meshHeight + 1); 
private Bitmap myBmp = 
BitmapFactory. decodeResource( getResources(), R.nmipmap. myimage); 
// 保 存 每 个 小 方 格 的 交叉 点 坐标 
private float[ ] myPoints = new float[myCount * 2]; 
Private float[ ] myVerts = new float[myCount * 2]; 
private float k; 
public myImageView(Context context, AttributeSet attrs, int defStyleAttr) { 
super(context, attrs, defStyleAttr); 
initData( ); 
} 
public myImageView(Context context, AttributeSet attrs){this(context, attrs, 0);} 
(Override 
protected void onDraw(Canvas myCanvas) { 
flagWave( ) ; 
k += FREQUENCY; 
myCanvas. translate(0,150); // 纵 向 平移 画布 
myCanvas. drawBitmapMesh( myBmp, 
meshWidth, meshHeight, myVerts, 0, null, 0, null]l); 
invalidate( ); 
} 
private void initData( ) { 
float myBmpWidth = myBmp. getWidth( ); 
float myBmpHeight = myBmp. getHeight( ); 
int index = 0; 
// 通 过 遍历 所 有 小 方 格 , 得 到 原 图 每 个 交叉 点 坐标 ,并 把 它们 保存 在 数组 中 
for (int i = 0; i<= meshHeight; i++) { 
float fy = myBmpHeight * i / meshHeight; 
for (intj = 0; j <= meshWidth; j++) { 
float fx = myBmpHeight * j / meshWidth; 
myPoints[index * 2 + 0] = myVerts[index * 2 + 0] 
myPoints[index x* 2 + 1] myVerts[index * 2 + 1] 
index++ ; 
六 守 ， 
// 加 入 正弦 函数 算法 ,修改 交叉 点 坐标 得 到 波纹 效果 ,这 里 只 修改 Y 坐 标 ,x 坐 标 不 变 
public void flagWave( ) { 
for (int i = 0; i<= meshHeight; i++) { 
for (int j = 0; j < meshWidth; j++) { 
myVerts[(i x (meshWidth + 1) + j) * 2 + 0] += 0; 
float offSetY = 
(float) Math. sin( (float) j / meshWidth * 2 * Math.PI + Math.PI * k); 
myVerts[(i * (meshWidth + 1) + j) * 2 + 1] = 
myPoints[(i * (meshWidth + 1) + j) * 2 + 1] + offSetY x* RMPLITUDE; 


Fx 
fy + AMPLITUDE * 1.2f; 


DE 


上 面 这 段 代 码 在 MyCode\ MySamplel46\app\src\ main\java\com\bin\luo\mysample\ 
myImageView. java 文件 中 。 在 这 段 代 码 中 ,drawBitmapMesh() 方 法 用 于 绘制 扭曲 图 像 ,该 方法 的 语 
法 声明 如 下 : 


第 4 章 ”图 形 和 图 像 《 


drawBitmapMesh ( (@ NonNull Bitmap bitmap, int meshWidth, int meshHeight, (@ NonNull float[ ] verts, int 
vertOffset, (Nullable int[ ] colors, int colorOffset, (@Nullable Paint paint) 


其 中 ,参数 Bitmap bitmap 表示 需要 扭曲 的 图 像 。 参 数 int meshWidth 表示 在 横向 上 把 该 图 像 划 
成 多 少 格 。 参 数 int meshHeight 表示 在 纵 回 上 把 该 图 像 划 成 多 少 格 。 参 数 float[L j verts 表示 长 度 为 
(meshWidth 十 1) * (meshHeight 十 1) * 2 的 数组 , 它 记 录 了 扭曲 后 的 图 像 各 顶点 位 置 。 参 数 int 
vertOffset 表示 verts 数组 中 从 第 几 个 数组 元 素 开 始 才 对 图 像 进行 扭曲 。 参 数 int| j colors 指定 每 个 
顶点 之 间 的 颜色 。 参 数 int colorOffset 表示 绘制 前 需要 跳 过 的 颜色 数 , 即 偏 移 量 。 参 数 Paint paint 表 
示 定 制 的 画笔 。 

此 实例 的 完整 项 目 在 MyCode\MySamplel146 文件 夹 中 ，。 


091 在 自 定 义 View 中 使 用 椭圆 裁剪 图 像 


此 实例 主要 通过 使 用 drawOval() 方 法 绘制 椭圆 ,并 在 setXfermode() 方 法 的 参数 中 使 用 android. 
graphics. PorterDuff. Mode. ADD 模式 创建 PorterDuffXfermode 实例 ,实现 将 图 像 裁 前 成 椭圆 。 当 实 
例 运行 之 后 ,在 自 定义 View 中 使 用 椭圆 裁剪 图 像 的 效果 如 图 091. 1 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
@ Override MySample 
protected void onCreate( Bundle savedInstanceState) { 
setContentView(new MyView(this)); 
super. onCreate( savedInstanceState) ; 
} 
class MyView extends View { 
public MYView(Context context) { super(context); } 


(Override 
protected void onDraw(Canvas myCanvas) { 
Display myDisplay = getWindowManager(). getDefaultDisplay(); 


int myWidth = myDisplay. getWidth(); 
int myHeight = myDisplay. getHeight( ); 
Bitmap myBitmap = 
BitmapFactory. decodeResource ( getResources ( ), R. mipmap. 


myimage); 
Paint myPaint = new Paint(); 091.1 


myCanvas. drawOval(10, myHeight / 5, myWidth - 10, 
myHeight * 3/ 5, myPaint); // 以 椭圆 形 裁剪 图 像 
myPaint. setXfermode(new PorterDuffXfermode(PorterDuff. Mode. ADD) ); 
myCanvas. drawBitmap(myBitmap, 0, 0, myPaint); // 绘 制图 像 
Pi 


上 面 这 段 代 码 在 MyCode\ MySample081 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , BitmapFactory. decodeResource (getResources(),R. 
mipmap. myimage) 用 于 从 R. mipmap. myimage 资源 中 获取 图 像 。myCanvas. drawOval (10， 
myHeight/5, myWidth 一 10,myHeight x* 3/5,myPaint) 用 于 绘制 椭圆 ,10 表示 左上 角 的 水 平 坐标 ( 因 
为 椭圆 内 切 于 和 矩形 ),myHeight/5 表示 左上 和 角 的 垂直 坐标 ,myWidth 一 10 表示 右 下 角 的 水 平 坐标 ， 
myHeight * 3/5 表示 右 下 角 的 垂直 坐标 ,myPaint 表示 男 笔 。、 myPaint， setXfermode (new 
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PorterDuffXfermode (PorterDuff. Mode. ADD)) 表 示 以 PorterDuff. Mode. ADD 模式 定制 画笔 , 当 在 
myCanvas. drawBitmap(myBitmap,0.0，myPaint) 中 使 用 此 画笔 绘制 图 像 时 , 则 仅 绘制 椭圆 部 分 的 对 
应 内 容 。 此 实例 的 完整 项 目 在 MyCode\MySample081 文件 夹 中 。 


092 ”通过 PorterDuff 模式 增 暗 显示 两 幅 图 像 


此 实例 主要 通过 设置 Paint 的 PorterDuffXfermode 为 DARKEN ,实现 仅 显 示 两 幅 图 像 颜 色 较 深 
的 部 分 。 当 实例 运行 之 后 , 单 击 “特效 图 像 ” 按 钮 , 则 两 幅 图 像 在 琶 加 之 后 仅 显 示 颜 色 较 深部 分 的 效果 
如 图 092. 1 的 左 图 所 示 。 单 击 “ 原 始 图 像 ” 按 钮 , 则 显示 原始 图 像 , 如 图 092. 1 的 右 图 所 示 。 


OA ~- 中 可 | ® x 二 I 1 


MySample 


图 092.1 
主要 代码 如 下 : 
public void onClickBtnl(View v) { // 响 应 单 击 "特效 图 像 "按钮 
Bitmap myBitmapl = 
BitmapFactory. decodeResource( getResources( ),R. mipmap. myimagel ); 
Bitmap myBitmap2 = 


BitmapFactory. decodeResource( getResources( ),R. mipmap. myimage2 ); 
Bitmap myBitmapCopy = myBitmap2. copy(Bitmap. Config. ARGB 8888, true); 
Canvas myCanvas = new Canvas (myBitmapCopy ); 

Paint myPaint = new Paint( ); 

// 获 得 每 个 位 置 上 两 幅 图 像 中 最 暗 的 像素 并 显示 

myPaint. setXfermode(new PorterDuffXfermode( PorterDuff. Mode. DARKEN ) ); 
myCanvas. drawBitmap(myBitmapl1,0,0,myPaint); 

myImageView. setImageBitmap(myBitmapCopy); 


上 面 这 段 代 码 在 MyCode\ MySample796\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myPaint. setXfermode (new PorterDuffXfermode 
(PorterDuff. Mode. DARKEN)) 中 的 DARKEN 表示 在 两 幅 图 像 琶 加 之 后 , 仅 显 示 颜 色 较 深 的 部 分 。 
此 实例 的 完整 项 目 在 MyCode\MySample796 文件 夹 中 。 


第 4 章 图形 和 图 像 


093 ”通过 PorterDuff 模式 将 图 像 裁 交 成 五 角 尾 


此 实例 主要 通过 使 用 PorterDuff. Mode. SRC_IN 相交 模式 琶 加 图 像 和 五 角 星 ,实现 将 图 像 裁剪 
成 五 角 星 形状 。 当 实例 运行 之 后 , 单 击 “ 显 示 原 图 ”按钮 , 则 在 屏 划 上 显示 原始 图 像 , 如 图 093. 1 的 左 
图 所 示 ; 单 击 “显示 五 星 图 ?按钮 , 则 将 以 五 角 星 形状 在 屏幕 上 显示 裁剪 后 的 图 像 , 如 图 093. 1 的 右 图 
所 示 。 


MySample 


显示 原 图 显示 五 星 图 


主要 代码 如 下 : 


public void onClickmyBtn2(View v) { // 响 应 单 击 " 显 示 五 星 图 "按钮 
myImageView. setImageBitmap( getNewImage( myBmp) ); 
} 
public Bitmap getNewImage(Bitmap myBitmap) { 
Bitmap myStarBmp = Bitmap. createBitmap(myWidth, myHeight, 
Bitmap. Config. ARGB 8888); // 创 建新 位 图 
Canvas myCanvas = new Canvas(myStarBmp); // 创 建 带 有 新 位 图 的 画布 
Paint myPaint = new Paint(); 
Path myPath = new Path!( ); 
myPath. moveTo(myWidth * 0.5f, myWidth * 0); 
myPath. lineTo(myWidth * 0.63f, myWidth * 0.38f); 
myPath. lineTo(myWidth, myWidth * 0.38f); 
myPath. lineTo(myWidth * 0.69f, myWidth * 0.59f); 
myPath. lineTo(myWidth * 0.82f, myWidth); 
myPath. lineTo(myWidth x* 0.5f, myWidth x* 0.75f); 
myPath. lineTo(myWidth * 0.18f, myWidth); 
myPath. lineTo(myWidth * 0.31f, myWidth * 0.59f); 
myPath. lineTo(0, myWidth * 0.38f); 
myPath. lineTo(myWidth * 0.37f, myWidth * 0.38f); 
myPath. close( ); 
myCanvas. translate(0,360); 
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myCanvas. drawPath(myPath, myPaint); // 绘 制 五 角 星 

// 设 置 相 交 模 式 

myPaint. setXfermode(new PorterDuffXfermodel(PorterDuff. Mode. SRC_IN) ); 
Rect myRect = new Rect(0, 0, myWidth, myWidth); // 把 图 像 画 到 五 星 
myCanvas. drawBitmap(myBitmap, null, myRect, myPaint); 

return myStarBmp; 


上 面 这 段 代 码 在 MyCode\ MySamplel08\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPath 表示 使 用 五 角 星 的 10 个 点 的 坐标 位 置 用 直线 连 
接 而 成 的 封闭 路 径 。myCanvas. drawPath (myPath，myPaint) 则 是 根据 封闭 路 径 绘 制 五 角 星 。 
myPaint. setXfermode (new PorterDuffXfermode (PorterDuf{f. Mode. SRC_IN)) 用 于 设置 夯 笔 的 
PorterDuffXfermode 模式 为 SRC_IN, 即 相交 模式 ,该 模式 规定 只 在 源 图 像 和 目标 图 像 相 交 的 地 方 绘 
制 源 图 像 。myCanvas. drawBitmap(myBitmap，null，myRect，myPaint) 则 表示 在 五 角 星 中 绘制 图 
像 。 此 实例 的 完整 项 目 在 MyCode\MySamplel108 文件 夹 中 。 


094 ”通过 PorterDuff 模式 改变 tint 属性 妥 加 效果 


此 实例 主要 通过 使 用 PorterDuff 的 多 种 图 像 处 理 模 式 , 使 tint 属性 指定 的 颜色 与 图 像 二 加 产生 
不 同 的 效果 。 当 实例 运行 之 后 , 单 击 “设置 ADD 模式 ”按钮 , 则 ImageView 控件 的 图 像 显 示 效 果 如 
图 094. 1 的 左 图 所 示 ; 单 击 “设置 SRC_OUT 模式 ”按钮 , 则 ImageView 控件 的 图 像 显 示 效 果 如 
图 094. 1 的 右 图 所 示 。 


MySample MySample 


设置 ADD 模 式 。 设置 SRC_OUT 模 式 ”|| 设置 ADD 模 式 ”设置 SRC_OUT 模 式 


图 094. 1 
主要 代码 如 下 : 
< ImageView android: id = "(@ + id/myImageView" 


androlid:layout_width = "wrap_content” 
android:layout height = "wrap_content" 
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android:layout centerInParent = "true" 
android:tint = "(Wandroid:color/holo green dark" 
android:tintMode = "multiply" 

android:src = "(mipmap/myimagel"/> 


上 面 这 段 代 码 在 MyCode\MySample561\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: tint 二 "(@android: color/holo_green_dark" 表 示 使 用 深 绿 色 与 图 像 进 行 羡 加 。 
android: tintMode 二 "multiply" 表 示 革 加 模式 是 PorterDuff. Mode. MULTIPLY。 在 Java 代码 中 , 则 
使 用 setImageTintMode() 方 法 设置 登 加 模式 ,主要 代码 如 下 : 


public void onClickBtnl(View v) { // 响 应 单 击 "设置 ADD 模式 "按钮 
myImageView. setImageTintMode( PorterDuff. Mode. ADD); 


} 

public void onClickBtn2(View v) { // 响 应 单 击 "设置 SRC_0UT 模式 "按钮 
myImageView. setImageTintMode(PorterDuff. Mode. SRC_ 0UT) ; 

} 


上 F 面 这 段 代 码 在 MyCode\ MySample561 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImageView. setImageTintMode(PorterDuff. Mode. 
ADD) 用 于 设置 ImageView 控件 的 tintMode 属性 。tintMode 属性 通常 支持 PorterDuff 列举 的 模式 ， 
如 CLEAR. SRC.DST.SRC OVER.DST OVER.SRC IN.DST IN SRC OUT.DST OUT SRC 
ATOP 、DST_ ATOP、. XOR.、 DARKEN.\ LIGHTEN.、 MULTIPLY.、 SCREEN.、 ADD、 OVERLAY 等 。 
此 实例 的 完整 项 目 在 MyCode\MySample561 文件 夹 中 。 


095 ”使 用 Region 的 DIFFERENCE 实现 抠 图 功能 


此 实例 主要 通过 使 用 Region. Op. DIFFERENCE 裁剪 模式 ,实现 在 图 像 上 抠 出 一 个 六 边 形 的 窟 
俐 。 当 实例 运行 之 后 , 单 击 “ 显 示 原 始 图 像 ” 按 钮 , 则 在 屏幕 上 显示 原始 图 像 ,如 图 095. 1 的 左 图 所 示 ; 
单 击 “ 显 示 抠 图 结果 ”按钮 , 则 原始 图 像 根 据 六 边 形 抠 图 之 后 的 结果 如 图 095. 1 的 右 图 所 示 。 
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显示 原始 图 像 显示 抠 图 结果 显示 原始 图 像 显示 抠 图 结果 
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主要 代码 如 下 : 


public void onClickBtn2(View v) { // 响 应 单 击 "显示 抠 图 结果 "按钮 
Bitmap myBitmap = BitmapFactory. decodeResource(getResources( ), 
R. mipmap. myimagel ). copy(Bitmap. Config. ARGB 8888, true); 
int myWidth = myBitmap. getWidth( ); 
int myYoffset = 250; 
Path myPath = new Path( ); 
myPath. moveTo(0, myWidth * 0.5f + myYoffset); 
myPath. lineTo(myWidth * 0.25f,0 + myYoffset); 
myPath. lineTo(myWidth * 0.75f, 0 + myYoffset); 
myPath. lineTo(myWidth , myWidth * 0.5f + myYoffset); 
myPath. lineTo(myWidth * 0.75f, myWidth + myYoffset); 
myPath. lineTo(myWidth x* 0.25f, myWidth + myYoffset); 
myPath. close( ); 
Region myRegion = new Region( ); 
myRegion. setPath(myPath, new Region(new Rect(0,0, 
myBitmap. getWidth( ), 
myBitmap. getHeight( ) ) ) ); // 根 据 封 闭 的 六 边 形 创建 Region 对 象 
Canvas myCanvas = new Canvas(myBitmap); 
myCanvas. drawColor( Color. WHITE); 
//myCanvas. clipRegion(myRegion, Region. Op. XOR); 
//Region. Op. DIFFERENCE 表示 获取 图 像 减 去 Region 的 剩余 部 分 , 即 抠 图 
myCanvas. clipRegion( myRegion, Region. Op. DIFFERENCE); 
myCanvas. drawBitmap( BitmapFactory. decodeResource(getResources( ), 
R. mipmap. myimagel),0,0,null); 
// 在 ImageView 上 显示 抠 图 (不 相交 的 部 分 ) 结 果 
myImageView. setImageBitmap(myBitmap); 


上 和 面 这 上段 代码 在 MyCode\ MySample828\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myCanvas. clipRegion (myRegion,， Region. Op. 
DIFFERENCE) 表 示 使 用 Region. Op. DIFFERENCE 模式 裁剪 图 像 , Region. Op. DIFFERENCE 模 
式 表示 裁剪 结果 是 全 集 减 去 交集 的 剩余 部 分 ( 即 差 值 )。 此 实例 的 完整 项 目 在 MyCode\\ 
MySample828 文件 夹 中 。 


096 ”使 用 ShapeDrawable 裁剪 三 角形 图 像 


此 实例 主要 以 三 角形 PathShape 创建 ShapeDrawable, 并 使 用 图 像 填 充 三 角形 风格 的 
ShapeDrawable ,实现 裁剪 三 角形 图 像 的 效果 。 当 实例 运行 之 后 , 单 击 “显示 原始 图 像 ” 按 钮 , 则 在 屏幕 
上 显示 原始 图 像 , 如 图 096. 1 的 左 图 所 示 ; 单 击 “ 显 示 三 角形 图 像 ” 按 钮 , 则 将 以 三 角形 在 屏幕 上 显示 
裁剪 的 图 像 , 如 图 096. 1 的 右 图 所 示 。 


public void onClickBtn2(View v) { // 响 应 单 击 " 显 示 三 角形 图 像 "按钮 
Bitmap myBitmap = 


BitmapFactory. decodeResource(getResources(), R.mipmap. myimagel ) ; 
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Path myPath = new Path( ); 
myPath. moveTo( myBitmap. getWidth( )/2,0); 
myPath. lineTo(myImageDrawable. getIntrinsicWidth( ), 
myImageDrawable. getIntrinsicHeight( )); 
myPath. lineTo(0,myImageDrawable. getIntrinsicHeight( )); 
myPath. close( ); // 根 据 三 角形 的 3 个 顶点 创建 封闭 三 角形 路 径 
ShapeDrawable myShapeDrawable = new ShapeDrawable(new PathShape(myPath, 
myImageDrawable. getIntrinsicWidth( ),myImageDrawable. getIntrinsicHeight() ) ); 
BitmapShader myBitmapShader = new BitmapShader(myBitmap，Shader.TileMode. CLAMP, 
Shader. TileMode. CLAMP); // 根 据 图 像 设置 对 应 BitmapShader 对 象 
myShapeDrawable. getPaint( ). setShader(myBitmapShader); // 应 用 该 BitmapShader 
myImageView. setBackground(myShapeDrawable); 


(| 
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显示 原始 图 像 显示 三 角形 图 像 显示 原始 图 像 显示 三 角形 图 像 


图 096.1 


上 F 面 这 段 代 码 在 MyCode\ MySample821 \app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPath 主要 用 于 创建 封闭 的 三 角形 路 径 。 
myShapeDrawable 主要 根据 myPath 创建 三 角形 ShapeDrawable。myBitmapShader 则 根据 图 像 创 建 
着 色 器 。myShapeDrawable. getPaint(). setShader(myBitmapShader) 则 使 用 该 图 像 着 色 器 填充 ( 绘 
制 ) 三 角形 的 内 部 ,实现 裁剪 三 角形 图 像 的 效果 。 此 实例 的 完整 项 目 在 MyCode\MySample821 文件 
夹 中 。 


097 ”使 用 ClipDrawable 裁剪 图 像 实现 星 级 评分 
此 实例 主要 通过 使 用 ClipDrawable 控制 裁剪 区 域 ,实现 星 级 评分 功能 。 当 实例 运行 之 后 , 单 击 


“增加 评分 ?按钮 , 则 一 次 增加 一 颗 星 ; 单 击 “减少 评分 ”按钮 , 则 一 次 减少 一 颗 星 ,效果 分 别 如 
图 097. 1 的 左 图 和 右 图 所 示 。 
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图 097. 1 


public class MainActivity extends Activity { 
ImageView myImageView; 
ClipDrawable myClipDrawable; 
int myLevel = 10000; 
Bitmap myBitmap; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
myBitmap = BitmapFactory. decodeResource(getResources(), R.mipmap.myscorestar); 
myClipDrawable = new ClipDrawable(new BitmapDrawable(myBitmap), 
Gravity. LEFT, ClipDrawable. HORIZONTAL ) ; 
myImageView. setImageDrawable( myClipDrawable); 
myClipDrawable. setLevel(myLevel ); 
} 
public void onClickBtnl (View v) { // 响 应 单 击 " 增 加 评分 "按钮 
if(myLevel < 10000){ myClipDrawable. setLevel(myLevel += 2000); } 
} 
public void onClickBtn2(View v) { // 响 应 单 击 "减少 评分 "按钮 
if(myLevel > 0){ myClipDrawable. setLevel(myLevel -=2000); } 
}} 


上 面 这 段 代 码 在 MyCode\ MySample782\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myDrawable. setLevel(myLevel 十 = 一 2000) 用 于 改变 裁剪 
区 域 , setLevel() 方 法 的 参数 取 值 范围 从 0 到 10000, 为 0 时 完全 不 显示 ,为 10000 时 完全 显示 ; 
10000/2000=5, 由 于 5 颗 星 是 一 幅 图 像 (灰色 的 五 星 也 是 一 幅 图 像 , 被 设置 为 背景 ), 因 此 每 增加 
2000 , 即 增 加 一 颗 。myDrawable = 二 new ClipDrawable(new BitmapDrawable (myBitmap) , Gravity. 
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LEFT,ClipDrawable. HORIZONTAL) 用 于 创建 指定 规则 的 ClipDrawable 对 象 ,此 实例 在 裁剪 图 像 
时 ,裁剪 规则 是 将 对 象 ( 图 像 ) 放 在 容 右 的 左 端 ,在 水 平方 呵 上 从 右 端 开始 裁剪 。 此 实例 的 完整 项 目 在 
MyCode\MySample782 文件 夹 中 。 


098 ”使 用 自 定义 Drawable 实现 对 图 像 进行 圆 角 


此 实例 主要 通过 使 用 Drawable 为 基 类 创建 自 定 义 类 RoundImageDrawable ,实现 对 ImageView 
控件 中 的 图 像 进行 圆 角 。 当 实例 运行 之 后 ,两 个 ImageView 控件 的 图 像 经 过 圆 角 之 后 的 效果 如 
图 098. 1 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { : 
ImageView mVImageView1l ; MySample 
ImageView myImageView2 ; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
Super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myImageView1l = (ImageView) findViewById(R. id. myImageViewl ) ; 
Bitmap myBmpl = 
BitmapFactory. decodeResourcel( getResources(),R. mipmap. myimagel ); 
myImageViewl. setImageDrawable(new RoundImageDrawable( myBmpl1, 30)); 
myImageView2 = (ImageView) findViewById(R. id.myImageView2); 
Bitmap myBmp2 = 
BitmapFactory. decodeResource( getResources( )，R. mipmap. myimage2 ) ; 
mylImageView2. setImageDrawable(new RoundImageDrawable(myBmp2, 30) ); 
} 
public class RoundImageDrawable extends Drawable{ 
private Paint myPaint; 098.1 
private Bitmap myBitmap; 
private RectF myRect; 


private float myRadius; 

public RoundImageDrawable(Bitmap bitmap, float radius) { 
myRadius = radius; 
myBitmap = bitmap; 
BitmapShader myBitmapShader = new BitmapShader (bitmap, 

Shader. TileMode. CLAMP, Shader. TileMode. CLAMP ) ; 

myPaint = new Paint(); 
myPaint. setAntiAlias(true); 
myPaint. setShader (myBitmapShader ); 

} 

(Override 

public void setBounds( int left, int top, int right, int bottom){ 
super. setBounds(left, top, right, bottom); 
myRect = new RectF(left, top, right, bottom); 

} 


(Override 

public void draw(Canvas canvas){ // 绘 制 圆 角 和 矩形 
canvas. drawRoundRect (myRect, myRadius, myRadius, myPaint); 

} 

(Override 


public int getIntrinsicWidth( ){ return myBitmap. getWidth( ); } 
(Override 


CNy Anaroia 引 本 应 局 300 。 实战 篇 


public int getIntrinsicHeight( ){ return myBitmap. getHeight( ); } 

人 Override 

public void setAlphal( int alpha){ myPaint. setAlpha(alpha); } 

(QOverride 

public void setColorFilter(ColorFilter cf){ myPaint. setColorFilter(cf); } 
(Override 

public int getOpacity( ){ return PixelFormat. TRANSLUCENT; } 

} 


上 面 这 段 代 码 在 MyCode\ MySample673\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,canvas. drawRoundRect(myRect, myRadius, myRadius,， 
myPaint) 用 于 根据 传递 的 圆 角 半径 myRadius 绘制 圆 角 矩形 , 即 对 图 像 进 行 圆 角 。 此 实例 的 完整 项 目 
在 MyCode\MySample673 文件 夹 中 ， 


099 ”使 用 Matrix 实现 按照 指定 方 品 倾斜 图 像 


此 实例 主要 通过 使 用 Matrix 的 postSkew() 方 法 倾斜 矩阵 ,实现 按照 一 定 的 方 回 倾斜 图 像 。 当 实 
例 运 行 之 后 , 单 击 “ 显 示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 ,如 图 099. 1 的 左 图 所 示 ; 单 击 “倾斜 图 
像 ” 按 钮 , 则 图 像 将 会 按照 指定 的 方 回 进行 倾斜 ,如 图 099. 1 的 右 图 所 示 。 


MySample 


图 099.1 
主要 代码 如 下 : 
public void onClickmyBtn2(View v) { // 响 应 单 击 " 倾 斜 图 像 "按钮 
myImageView. setImageBitmap(getNewImage(myBitmap) ) ; 
} 


public Bitmap getNewImage(Bitmap oldBmp) { 
int myWidth = oldBmp.getNWidth( ) ; 
int myHeight = oldBmp. getHeight( ) ; 
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Matrix myMatrix = new Matrix(); 

//myMatrix. preSkew(0.1f, 0.2f); 

myMatrix. postSkew(0.1f, 0.2f); // 倾 和 斜 图 像 

Bitmap newBmp = Bitmap. createBitmap(oldBmp, 0, 0, myWidth, 
myHeight，myMatrix， true); // 根 据 原始 图 像 和 倾斜 之 后 的 矩阵 创建 新 图 像 


return newBmp; 


上 面 这 上段 代码 在 MyCode\ MySamplel27\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myMatrix. postSkew (0. 1f，0. 2f) 表 示 将 myMatrix 和 矩阵 
相对 于 工 轴 倾 斜 0. 1f, 相 对 于 > 轴 倾 斜 0. 2f, 此 处 的 倾斜 理解 为 扭曲 可 能 更 合适 些 。Bitmap. 
createBitmap(oldBmp, 0, 0, myWidth, myHeight, myMatrix, true) 用 于 根据 oldBmp 图 像 创 建 与 
倾斜 之 后 的 myMatrix 相 匹 配 的 新 图 像 。 此 实例 的 完整 项 目 在 MyCode\MySamplel127 文件 夹 中 。 


100 ”使 用 ColorMatrix 为 图 像 添 加 所 紫 效果 


此 实例 主要 通过 直接 使 用 ColorMatrix 颜色 和 矩阵 配置 画笔 ,实现 为 图 像 添 加 泛 紫 效果 。 当 实例 运 
行 之 后 , 单 击 “显示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 , 如 图 100. 1 的 左 图 所 示 ; 单 击 “ 泛 紫 图 像 ” 按 
钮 , 则 显示 泛 紫 效果 的 图 像 , 如 图 100. 1 的 右 图 所 示 。 


主要 代码 如 下 : 


public void onClickmyBtn2(View v) { // 响 应 单 击 " 泛 紫 图 像 "按钮 
Bitmap newImage = getNewImage(myBitmap); 
myImageView. setImageBitmap(newImage); 

} 

public Bitmap getNewImage(Bitmap oldBmp) { 
int myWidth = oldBmp. getWidth( ); 
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int myHeight = oldBmp. getHeight(); 
float[ ] myColorArray = {1, 0, 11, 0, 10, 
G1 50.0. 100, 
00. 1 0 0 
1 PR // 泛 紫 颜 色 和 矩阵 
Bitmap newBmp = Bitmap. createBitmap(myWidth, 
myHeight, Bitmap. Config. ARGB 8888 ) ; 
Canvas myCanvas = new Canvas(newBmp); 
Paint myPaint = new Paint(); 


ColorMatrix myColorMatrix = new ColorMatrix( ); // 新 建 颜 色 和 矩阵 
myColorMatrix. set(myColorArray); // 设 置 泛 紫 颜 色 和 矩阵 
// 通 过 颜色 过 滤器 配置 泛 紫 画 笔 

myPaint. setColorFilter(new ColorMatrixColorFilter(myColorMatrix) ); 
myCanvas. drawBitmap(oldBmp, 0, 0, myPaint); // 绘 制 泛 紫 图 像 
return newBmp; 


上 面 这 上段 代码 在 MyCode\ MySamplel33\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,floatl ] myColorArray = {1, 0, 11, 0, 10, 0, 1, 0, 0， 
100, 0，100，1，0,， 0， 0，0，0，1,， 0 是 泛 紫 颜色 和 窍 阵 的 数组 表达 式 ， myColorMatrix. set 
(myColorArray) 用 于 实现 在 使 用 该 数组 构建 颜色 矩阵 与 图 像 在 画布 上 进行 绘制 时 ( 即 myCanvas. 
drawBitmap(oldBmp，0，0，myPaint) ) ,将 使 图 像 产生 泛 紫 效果 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample133 文件 夹 中 。 


101 ”使 用 ColorMatrix 实现 图 像 的 加 暗 效 果 


此 实例 主要 通过 直接 使 用 ColorMatrix 颜色 矩阵 配置 画笔 ,实现 为 图 像 添 加 加 暗 效果 。 当 实例 运 
行 之 后 , 单 击 " 显 示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 ,如 图 101. 1 的 左 图 所 示 ; 单 击 “ 加 上 暗 图 像 ” 按 
钮 , 则 显示 加 暗 效果 的 图 像 ,如 图 101. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickmyBtn2(View v) { // 响 应 单 击 " 加 暗 图 像 " 按 钮 
Bitmap newImage = getNewImage(myBitmap); 
myImageView. setImageBitmap(newImage); 
} 
public Bitmap getNewImage(Bitmap oldBmp) { 
int myWidth = oldBmp. getWidth(); 
int myHeight = oldBmp. getHeight(); 
float[ ] myColorArray = {1,0,0,0, - 100, 
-1 0 0 100. 
D00205 = i100 
0,0,0,1,0}); // 加 暗 颜色 矩阵 
Bitmap newBmp = Bitmap. createBitmap(myWidth, 
myHeight, Bitmap. Config.ARGB 8888); 
Canvas myCanvas = new Canvas(newBmp); 
Paint myPaint = new Paint(); 
ColorMatrix myColorMatrix = new ColorMatrix();  // 新 建 颜色 和 矩阵 
myColorMatrix. set(myColorArray); // 设 置 加 暗 颜 色 和 矩阵 
// 通 过 颜色 过 滤器 配置 加 暗 画 笔 
myPaint. setColorFilter(new ColorMatrixColorFilter(myColorMatrix) ); 
myCanvas. drawBitmap( oldBmp, 0, 0, myPaint); // 绘 制 加 暗 图 像 
return newBmp; 


上 面 这 段 代 码 在 MyCode\ MySamplel41\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,floatl | myColorArray 二 {1,0,0,0, 一 100, 0,1, 0,0, 一 
100,0,0,1,0, 一 100.0,0.0,1,0} 是 加 暗 图 像 的 颜色 和 罕 阵 的 数组 表达 式 ,myColorMatrix。 set 
(myColorArray) 用 于 实现 在 使 用 该 数组 构建 颜色 和 矩阵 与 图 像 在 画布 上 进行 绘制 时 ( 即 myCanvas. 
drawBitmap(oldBmp，0，0，myPaint)), 将 使 图 像 产 生 加 暗 效 果 。 此 实例 的 完整 项 目 在 MyCode\ 
MySamplel41 文件 夹 中 ， 


102 ”通过 自 定义 ColorMatrix 调整 图 像 赣 色色 调 


此 实例 主要 通过 直接 使 用 ColorMatrix 颜色 和 矩阵 配置 画笔 ,实现 使 用 蓝 色 通道 过 滤 图 像 。 当 实例 
运行 之 后 , 单 击 “ 显 示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 ,如 图 102. 1 的 左 图 所 示 ; 单 击 “使 用 蓝 色 
通道 过 滤 图 像 ” 按 钮 , 则 图 像 在 经 过 蓝 色 通 道 过 滤 之 后 的 效果 如 图 102. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


// 响 应 单 击 "使 用 蓝 色 通 道 过 滤 图 像 " 按 钮 
public void onClickmyBtn2(View v) { 
Bitmap newImage = getNewImage(myBitmap); 
myImageView. setImageBitmap(newImage); 
} 
public Bitmap getNewImage(Bitmap oldBmp) { 
int myWidth = oldBmp. getWidth( ); 
int myHeight = oldBmp. getHeight( ); 
// 定 义 蓝 色 通 道 颜色 矩阵 数组 值 
float[ ] myColorArray = { 
DERD 0 O00 
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Bitmap. createBitmap(myWidth, 
myHeight, Bitmap. Config. ARGB 8888); 
Canvas myCanvas = new Canvas(newBmp); 
Paint myPaint = new Paint( ); 
// 新 建 颜 色 和 矩阵 
ColorMatrix myColorMatrix = new ColorMatrix( ) ; 
// 设 置 蓝 色 通 道 颜色 和 矩阵 
myColorMatrix. set(myColorArray); 
// 通 过 颜色 过 滤器 配置 画笔 
myPaint. setColorFilter(new ColorMatrixColorFilter(myColorMatrix) ); 
// 绘 制 蓝 色 通 道 过 滤 的 图 像 
myCanvas. drawBitmap(oldBmp, 0, 0, myPaint); 
return newBmp; 
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图 102.1 


上 面 这 段 代 码 在 MyCode\ MySample417\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,floatl | myColorArray 二 {0,0,0,0,0,0,0,0,0,0,0,0,1， 
0.0,0.0.0,1.0} 是 蓝 色 通道 颜色 和 矩阵 的 数组 表达 式 ,myColorMatrix. set(myColorArray) 用 于 在 使 用 
该 数组 构建 颜色 怎 阵 与 图 像 在 画布 上 进行 绘制 时 ( 即 myCanvas. drawBitmap (oldBmp, 0, 0， 
myPaint) ) ,将 使 图 像 仅 显示 蓝 色 通道 支持 的 部 分 。 此 实例 的 完整 项 目 在 MyCode\MySample417 文 
件 夹 中 ，。 


103 ”使 用 RenderScript 实现 高 斯 算法 模糊 图 像 


此 实例 主要 通过 使 用 RenderScript 和 ScriptIntrinsicBlur ,实现 以 高 斯 模式 模糊 图 像 。 当 实例 运 
行 之 后 , 单 击 “ 显 示 原 始 图 像 ” 按 钮 , 则 在 屏 莫 上 显示 原始 图 像 ,如 图 103. 1 的 左 图 所 示 ; 单 击 “高 斯 模 


糊 图 像 ” 按 钮 , 则 显示 经 过 模糊 处 理 之 后 的 图 像 , 如 图 103. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


// 响 应 单 击 "高 斯 模糊 图 像 "按钮 
public void onClickmyBtn2(View v) { 
myBitmap = BitmapFactory. decodeResource(getResources( ) ，R. mipmap.myimagel ) ; 
Bitmap myBitmap = blurBitmap(this.myBitmap, this); 
myImageView. setImageBitmap(myBitmap); 
} 
public static Bitmap blurBitmap(Bitmap bitmap, Context context) { 
Bitmap outBitmap = Bitmap. createBitmap(bitmap.getWidth( ), 
bitmap. getHeight(), 
Bitmap. Config. ARGB 8888); 
// 初 始 化 Renderscript, 这 个 类 提供 了 RenderScript 上 下 文 ,在 创建 其 他 RS 类 
// 之 前 必须 要 先 创 建 这 个 类 ,以 控制 RenderScript 的 初始 化 .资源 管理 、 释 放 
RenderScript myRenderScript = RenderScript. create(context); 
ScriptIntrinsicBlur myBlurScript = ScriptIntrinsicBlur. create( 


myRenderScript, Element.U8 4(myRenderScript)); // 创 建 高 斯 模糊 对 象 
// 创 建 Allocations, 此 类 是 将 数据 传递 给 RenderScript 内 核 的 主要 方法 ， 
// 并 制定 后 备 类 型 存储 给 定 类 型 


Allocation myIn = Allocation.createFromBitmap(myRenderScript, bitmap); 
Allocation myOut = Allocation.createFromBitmap(myRenderScript, outBitmap); 
// 设 定 模 糊 度 

myBlurScript. setRadius(15.f£); 

// 执 行 Renderscript 

myBlurScript. setInput(myIn); 

myBlurScript. forEach( myOut ); 

myOut. copyTo( outBitmap); 

bitmap. recycle( ); 

myRenderScript. destroy( ); 

return outBitmap; 
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上 F 面 这 段 代 码 在 MyCode\ MySample661\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample661 文件 夹 中 。 


104 ”使 用 拉 普 拉 斯 模板 实现 图 像 的 锐 化 特效 


此 实例 主要 实现 了 使 用 拉 普 拉 斯 模板 对 图 像 进行 锐 化 特效 处 理 。 锐 化 图 像 实 质 上 就 是 要 突出 图 
像 中 有 关 形体 的 边缘 。 所 谓 形体 的 边缘 就 是 图 像 像素 点 的 颜色 值 发生 显 车 变化 的 地 方 , 在 图 像 的 平 
淡 区 ,这 种 颜色 值 的 变化 比较 平缓 ,而 在 图 像 的 边缘 区 域 这 种 变化 则 相当 明显 。 当 实例 运行 之 后 , 单 
击 “ 显 示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 , 如 图 104. 1 的 左 图 所 示 ; 单 击 “ 显 示 锐 化 图 像 ” 按 钮 , 则 
将 显示 经 过 锐 化 处 理 之 后 的 图 像 , 如 图 104. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


// 响 应 单 击 "显示 锐 化 图 像 " 按 钮 
public void onClickmyBtn2(View v) { 


// 第 一 次 锐 化 图 像 
Bitmap myShowBmp = getNewImage( myBitmap); 
// 第 二 次 锐 化 图 像 
myShowBmp = getNewImage( myShowBmp); 
// 显 示 锐 化 之 后 的 图 像 
myImageView. setImageBitmap(myShowBmp); 
} 
public Bitmap getNewImage(Bitmap myBitmap) { 
// 定 义 拉 普 拉 斯 矩阵 数组 值 
int[ ] myLaplacian = new int[]{-1, -1, -1, 
me 
ee 


int myWidth = myBitmap. getWidth( ); 
int myHeight = myBitmap. getHeight( ); 
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int oldR = 0,oldG = 0,oldB = 0; 
int[ ] myPixels = new int[myWidth * myHeight]; 
myBitmap. getPixels(myPixels, 0, myWidth, 0, 0, myWidth, myHeight); 
for (int i = 1; i< myHeight - 1; i++) { 
for (int j = 1; j < myWidth- 1; j++) { 
int myPixel = 0; 
int newR = 0; 
int newG = 0; 
int newB = 0; 
int myIndex = 0; 
for (intm = -1;m<= 1; mt+) { 
for (intn = -1;n<= 1; n++){ 
myPixel = myPixels[(i + n)* myWidth + (j + m)]; 
oldR = Color.red(myPixel); 
oldG = Color.green(myPixel ); 
oldB = Color.blue(myPixel); 
newR = newR + oldR * myLaplacian[myIndex]; 
newG = newG + oldG * myLaplacian[myIndex]; 
newB = newB + oldB * myLaplacian[myIndex]; 
myIndext++; 
大 
newR = Math.min(255, Math.max(0, newR)); 
newG = Math.min(255, Math.max(0, newG)); 
newB = Math.min(255, Math.max(0, newB)); 
myPixels[(i-1) * myWidth + (j-1)] = Color.argb(255,newR, newG, newB); 
} } 
// 根 据 原始 图 像 创建 与 其 尺寸 完全 相同 的 新 图 像 
Bitmap newBmp = Bitmap. createBitmap(myWidth, 
myHeight, Bitmap. Config.RGB 565); 
// 根 据 新 的 像素 值 填充 新 图 像 
newBmp. setPixels(myPixels, 0, myWidth, 0, 0, myWidth, myHeight); 
return newBmp; 


上 面 这 上段 代码 在 MyCode\ MySamplel22\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,int[ ] myLaplacian = new int[ {一 1, 一 1, 一 1, 一 1, 9， 
一 1, 一], 一 1, 一 1) 即 是 拉 普 拉 斯 模板 的 数组 表示 方式 。myBitmap. getPixels (myPixels，0， 
myWidth, 0, 0, myWidth, myHeight) 用 于 获取 原始 图 像 的 像素 值 ,并 保存 在 数组 myPixels 中 。 
newR = newR 十 oldR * myLaplacian | myIndex | newG = newG 十 oldG * myLaplacian 
[myIndex] 和 newB = newB 十 oldB x myLaplacian[ myIndexj] 即 是 根据 拉 普 拉 斯 模板 对 厚 像 素 的 
RGB 分 量 进行 数学 运算 从 而 产生 新 的 像素 以 形成 整 幅 图 像 的 锐 化 效果 。newR = Math. min(255， 
Math. max(0, newR)) .newG = Math. min(255, Math. max(0, newG)) 和 newB = Math. min(255 ， 
Math. max(0, newB)) 用 于 处 理 R、G.、B 分 量 溢出 的 情况 , 即 禁 止 R.G.、B 分 量 出 现 小 于 0 或 大 于 255 
的 情况 。newBmp. setPixels(myPixels, 0, myWidth, 0, 0, myWidth, myHeight) 用 于 根据 处 理 之 后 
的 像素 值 在 空白 图 像 中 填充 像素 , 即 生成 锐 化 图 像 。 此 实例 的 完整 项 目 在 MyCode\MySample122 文 
件 夹 中 。 
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105 ”通过 像素 操作 实现 在 图 像 上 添加 光照 效果 


此 实例 主要 通过 使 用 getPixels() 方 法 和 setPixels() 方 法 操作 图 像 的 像素 ,实现 在 图 像 上 添加 光 
照 效果 。 当 实例 运行 之 后 , 单 击 “ 显 示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 ,如 图 105. 1 的 左 图 所 示 ; 
单 击 “ 显 示 光 照 图像 ” 按 钮 , 则 将 显示 中 心 光 照 的 图 像 ,如 图 105. 1 的 右 图 所 示 。 


图 105.1 
主要 代码 如 下 : 
public void onClickmyBtn2(View v) { // 响 应 单 击 " 显 示 光 照 图 像 "按钮 
myImageView. setImageBitmap(getNewImage(myBitmap) ) ; 
I 


public Bitmap getNewImage(Bitmap oldBmp) { 
final int myWidth = oldBmp. getWidth( ); 
final int myHeight = oldBmp. getHeight(); 
int centerX = myWidth / 2; 
int centerY = myHeight / 2; 
int myRadius = Math.min(centerX, centerY); // 设 置 光 照 半 径 
final float myLight = 150F; // 设 置 光 照 强 度 
int[ ] myPixels = new int[myWidth x myHeight]; 
oldBmp. getPixels(myPixels, 0, myWidth, 0, 0, myWidth, myHeight); 
int myPos = 0; 
for (int i = 1, length = myHeight -~ 1; i< length; i++) { 
for (int j = 1, len = myWidth - 1; j < len; j++) { 
myPos = i * myWidth + j; 
int oldR = 0, oldG = 0, oldB = 0, myPixel = 0; 
int newR = 0, newG = 0, newB 0; 
myPixel = myPixels[myPos]; 
oldR = Color.red(myPixel); 
oldG = Color.green(myPixel ); 
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oldB = Color.blue(myPixel); 


newR = oldR; 
newG = oldG; 
newB = oldB; 


// 计 算 当 前 点 到 中 心 的 距离 ,在 平面 坐标 系 中 求 两 点 之 间 的 距离 

int myLength = (int) (Math. Pow((centerY - i), 2) + Math. pow(centerX - ]j, 2)); 
if (myLength < myRadius * myRadius) { 

// 按 照 距离 大 小 计算 增加 的 光照 强度 

int myDifference = (int) (myLight * (1.0 - Math. sqrt(myLength) / myRadius) ) ; 
newR = oldR + myDifference; 


newG = oldG + myDifference; 

newB = oldB + myDifference; 

} 

newR = Math.min(255, Math.max(0, newR)); 
newG = Math.min(255, Math.max(0, newG)); 
newB = Math.min(255, Math.max(0, newB)); 


myPixels[myPos] = Color.argb(255, newR, newG, newB); 
}} 
// 根 据 原始 图 像 创建 与 其 尺寸 完全 相同 的 新 图 像 
Bitmap newBmp = Bitmap.createBitmap(myWidth, myHeight, Bitmap.Config.RGB 565); 


// 根 据 新 的 像素 值 填 充 新 图 像 
newBmp. setPixels(myPixels, 0, myWidth, 0, 0, myWidth, myHeight ); 
return newBmp; 


上 面 这 段 代 码 在 MyCode\ MySamplel23\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , oldBmp. getPixels (myPixels，0，myWidth，0，0， 
myWidth, myHeight) 用 于 获取 厚 始 图 像 的 像素 值 , 并 保存 在 数组 myPixels 中 。newBmp. setPixels 
(myPixels, 0, myWidth, 0, 0, myWidth, myHeight) 用 于 根据 添加 中 心 光 照 之 后 的 像素 值 在 空白 
图 像 中 填充 像素 。newR = oldR 十 myDifference ,newG 三 oldG 十 myDifference 和 newB = oldB 
十 myDifference 用 于 给 指定 范围 的 像素 添加 光照 。newR = Math. min (255，Math. max (0， 
newR)) .newG = Math. min(255, Math. max(0, newG)) 和 newB = Math. min(255, Math. max 
(0, newB)) 用 于 处 理 RG、B 分 量 溢出 。 因 为 当 应 用 上 面 的 数学 算法 重 置 R.G、B 分 量 时 ,很 可 能 产 
生 RG、B 分 量 大 于 255 的 情况 ; 这 种 情况 在 像素 值 中 是 不 可 能 存在 的 ,所 以 一 旦 发 生 这 种 情况 , 则 将 
其 强行 设置 为 最 大 值 255。 此 实例 的 完整 项 目 在 MyCode\MySamplel23 文件 夹 中 。 


106 ”通过 像 系 操作 使 彩色 图 像 呈 现 浮 雕 特效 


此 实例 主要 通过 使 用 getPixels() 方 法 和 setPixels() 方 法 操作 图 像 的 像素 ,实现 将 彩色 图 像 转 化 
为 浮雕 图 像 。 当 实例 运行 之 后 , 单 击 “ 显 示 原 图 ”按钮 , 则 在 屏幕 上 显示 原始 图 像 ,如 图 106. 1 的 左 图 
所 示 ; 单 击 “ 显 示 浮 雕 图 像 ” 按 钮 , 则 将 显示 浮雕 效果 的 图 像 ,如 图 106. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickmyBtn2(View v) { // 响 应 单 击 " 显 示 浮 雕 图 像 " 按 钮 
int myWidth = myBitmap. getWidth(); 
int myHeight = myBitmap. getHeight(); 
int myPixel = 0, prePixel = 0, a, r, 9 b; 
int r1, gl, bl; 
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int[ ] oldPixels new int[myWidth x* myHeight]; 
int[ ] newPixels = new int[myWidth * myHeight]; 
// 获 取 原 始 图 像 每 个 点 的 像素 值 
myBitmap. getPixels(oldPixels, 0, myWidth, 0, 0, myWidth, myHeight); 
for (int i = 1; i< oldPixels. length; i++) { 
prePixel = oldPixels[i - 1]; 
a = Color.alpha(prePixel); 


Color. red(prePixel); 
g = Color.green(prePixel); 
b = Color.bluel(prePixel); 
myPixel = oldPixels[i]; 

rl = Color.red(myPixel); 
gl = Color.green(myPixel); 


bl = Color.blue(myPixel); 
r= r= £1+ 127; 
y= gl gt 127 
b= bl ht 127; // 处 理 相 邻 两 个 像素 点 的 RGB 分 量 ,使 之 产生 浮雕 效果 
newPixels[i] = Color.argb(a,r > 255 ? 255 : r, 

g>255? 255 : g, b> 255 ? 255 :b ); // 处 理 RGB 值 超过 255 情形 
} 
Bitmap newBmp = Bitmap. createBitmap(myWidth, myHeight, 

Bitmap. Config. ARGB 8888); // 根 据 原 始 图 像 创 建 与 其 尺寸 完全 相同 的 新 图 像 
newBmp. setPixels(newPixels, 0, myWidth, 0, 0, 

myWidth，myHeight); // 根 据 新 的 像素 值 填充 新 图 像 

myImageView. setImageBitmap( newBmp); 
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上 面 这 段 代 码 在 MyCode\ MySamplel21\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myBitmap。getPixels (oldPixels，0，myWidth，0，0， 
myWidth, myHeight) 用 于 获取 原始 图 像 的 像素 值 , 并 保存 在 数组 中 。r 二 rl 一 r 十 127.g 二 gl 一 
g 十 127 和 b= 二 bl 一 b 十 127 表示 将 相 邻 像素 点 的 R、G、B 分 量 相 减 后 加 127, 从 而 使 整个 图 像 产 
生 浮 雕 效 果 。newPixels[i| ==Color. argb(a,r > 255 ? 255 : r, g > 255 ? 255 :; g, b > 255 ? 255 : b) 
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用 于 处 理 R、.G、B 分 量 溢出 的 情况 。newBmp. setPixels(newPixels, 0, myWidth, 0, 0，myWidth ， 
myHeight) 用 于 根据 处 理 之 后 的 像素 值 在 空白 图 像 中 填充 像素 , 即 生成 的 浮雕 图 像 。 此 实例 的 完整 
项 目 在 MyCode\MySamplel21 文件 夹 中 。 


107 使 用 BitmapShader 实现 文字 线条 图 像 化 


此 实例 主要 通过 使 用 BitmapShader 自 定义 Paint 对 象 ,实现 使 用 图 像 填 充 文 字 线 条 。 当 实例 运 
行 之 后 , 单 击 “ 图 像 文字 ”按钮 , 则 使 用 图 像 填 充 文字 线条 ,效果 如 图 107. 1 的 左 图 所 示 。 单 击 “ 渐 变 文 
字 ” 按 钮 , 则 使 用 线性 渐变 色 填 充 文字 线条 的 效果 ,如 图 107. 1 的 右 图 所 示 。 
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图 107.1 
主要 代码 如 下 : 
public void onClickBtnl(View v) { // 响 应 单 击 "图 像 文字 "按钮 
Bitmap myBackground = 


BitmapFactory. decodeResource(getResources( )，R.mipmap.myimagel); 
drawTextWithEffect(" 案 " ，myBackground，nul1); 
} 
public void onClickBtn2(View v) { // 响 应 单 击 "渐变 文字 "按钮 
LinearGradient myGradient = new LinearGradient(0, 0, 1080, 0, Color.RED, 
Color. GREEN, Shader.TileMode.REPEAT); // 创 建 从 红色 至 绿色 的 渐变 对 象 
drawTextWithEffect(" 案 ", null, myGradient ); 
} 
public void drawTextWithEffect(String text, 
Bitmap bitmap, LinearGradient gradient){ 
Bitmap myBitmap = Bitmap. createBitmap( 
getWindowManager( ) .getDefaultDisplay().getWidth(), 
getWindowManager( ) .getDefaultDisplay( ).getHeight( ), 
Bitmap. Config. ARGB 8888); 
Canvas myCanvas = new Canvas(myBitmap); 
Paint myPaint = new Paint(); 
myPaint. setTextSize(1080); // 设 置 字体 大 小 
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} 


myPaint. setTypeface( Typeface. DEFAULT BOLD); // 加 粗 文字 

if (bitmap == null) { // 使 用 渐变 色 填 充 文字 
myPaint. setShader(gradient); 

} else if (gradient == null) { // 使 用 图 像 填 充 文字 


myPaint. setShader (new BitmapShader( bitmap, 
Shader. TileMode. REPEAT, Shader. TileMode. REPEAT) ) ; 
} 
myCanvas. drawText(text, 0, 1200, myPaint); 
myImageView. setImageBitmap(myBitmap); 


上 面 这 上段 代码 在 MyCode\ MySample790\app\src\ main\java\com\bin\luo\mysample\ 


MainActivity. java 文件 中 。 在 这 段 代 码 中 , new BitmapShader (bitmap,， Shader. TileMode. 
REPEAT,Shader. TileMode. REPEAT) 用 于 创建 位 图 浓 染 融 , 在 Paint 中 使 用 该 浑 染 器 来 实现 使 用 
图 像 填充 文字 线条 。BitmapShader() 方 法 的 语法 声明 如 下 : 


public BitmapShader( (NonNull Bitmap bitmap, TileMode tilex, TileMode tileY) 


其 中 ,参数 Bitmap bitmap 表示 在 演 染 器 内 呈现 的 图 像 。 参 数 TileMode tileX 表示 在 位 图 的 工 方 回 
的 平 铺 模式 。 参 数 TileMode tileY 表示 在 位 图 的 y 方向 的 平 铺 模式 。TileMode 一 共有 如 下 3 种 类 型 。 

(1) CLAMP, 如 果 演 染 需 超出 厚 始 边界 范围 , 则 复制 范围 内 边缘 的 颜色 进行 泻 染 。 

(2) REPEAT, 在 横向 和 纵向 重复 图 像 , 平 铺 。 

(3) MIRROR ,在 横向 和 纵向 重复 图 像 , 以 镜像 方式 平 铺 。 

此 实例 的 完整 项 目 在 MyCode\MySample790 文件 夹 中 。 


108 ”使 用 BlurMaskFilter 为 图 像 添 加 轮廓 线 


此 实例 主要 通过 使 用 BlurMaskFilter 定制 阴影 画笔 ,同时 为 画笔 设置 颜色 ,实现 为 png 格式 的 图 
像 添 加 轮廓 线 。 当 实例 运行 之 后 , 单 击 “ 添 加 特效 ?按钮 , 则 在 图 像 的 轮廓 上 出 现 一 条 红色 的 线条 , 效 


果 如 图 108. 1 的 左 图 所 示 。 单 击 “ 移 除 特效 ”按钮 , 则 将 显 


示 原 始 图 像 ,如 图 108. 1 的 右 图 所 示 。 
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public void onClickBtn1 (View v) { // 响 应 单 击 "添加 特效 "按钮 
Bitmap myBitmap = 
BitmapFactory. decodeResource( getResources(),R.mipmap. myimagel ); 
// 初 始 化 外 阴影 遮 醒 过 滤器 


BlurMaskFilter myFilter = new BlurMaskFilter(10,BlurMaskFilter.Blur. OUTER); 
Paint myPaint = new Paint( ); 


myPaint. setColor( Color. RED); // 设 置 红色 画笔 
myPaint. setMaskFilter(myFilter); // 在 画笔 中 应 用 模糊 过 滤器 
int[ ] myOffset = new int[2]; 
Bitmap myImage = myBitmap.extractRlpha( ); // 生 成 图 像 底 图 
Bitmap myImageCopy = 
myImage. copy(Bitmap. Config. ARGB 8888,true); // 生 成 底 图 副本 
Canvas myCanvas = new Canvas(myImageCopy); // 根 据 该 副本 初始 化 画布 
myCanvas. drawBitmap(myImage, 0,0, myPaint); // 绘 制 底 图 和 阴影 
myCanvas. drawBitmap(myBitmap, - myOffset[0], 
-~ myOffset[1],nul]l); // 在 底 图 上 覆盖 绘制 原 图 
myImageView. setImageBitmap(myImageCopy); // 显 示 特 效 
} 
public void onClickBtn2(View v) { // 响 应 单 击 " 移 除 特 效 " 按 钮 


Bitmap myBitmap = 
BitmapFactory. decodeResource( getResources(), R.mipmap. myimagel ); 
myImageView. setImageBitmap(myBitmap); 
} 


上 面 这 段 代 码 在 MyCode\ MySample789\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 , BlurMaskFilter myFilter 二 new BlurMaskFilter (10， 
BlurMaskFilter. Blur. OUTER) 用 于 创建 半径 为 10 的 外 阴影 过 滤器 ,由 于 外 阴影 半径 较 小 , 它 实际 看 
起 来 就 像 是 一 根 线条 。myPaint. setColor(Color. RED) 用 于 设置 画笔 颜色 ,该 颜色 将 作用 于 外 阴影 六 
滤器 ,因此 可 以 据 此 设置 不 同 的 阴影 颜色 。 此 实例 的 完整 项 目 在 MyCode\MySample789 文件 夹 中 。 


109 ”使 用 PathDashPathEffect 实现 柏 圆 线条 


此 实例 主要 通过 使 用 PathDashPathEffect 创建 路 径 特 效 ,实现 使 用 椭圆 填充 线条 。 当 实例 运行 
之 后 ,使 用 椭圆 线 条 绘制 的 椭圆 效果 如 图 109. 1 所 示 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { -A 
(Override MyGumpie 
protected void onCreate( Bundle savedInstanceState) { 

setContentView(new MyView(this)); 
super. onCreate(savedInstanceState) ; 
} 
class MyView extends View { 
public MyView(Context context) { super(context); } 
(WOverride 
protected void onDraw(Canvas myCanvas) { 
Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay. getWidth(); 
int myHeight = myDisplay. getHeight( ); 
Path myOval = new Path( ); 
myOval.addOval(0, 0, 46, 28, Path.Direction.CCW); 
PathEffect myEffect = new PathDashPathEffect(myOval, 
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Paint myPaint = new Paint(); 
myPaint. setColor(Color. BLUE); 
myPaint. setAntiAlias(true); 
myPaint. setPathEffect (myEffect); // 设 置 椭圆 路 径 特效 
myCanvas. drawOval (myWidth/10, myHeight/20, 
myWidth - myWidth/10,myHeight - myHeight/6,myPaint); ”// 绘 制 大 椭圆 


48,0, PathDashPathEffect. Style. MORPH); 


Pt 


上 面 这 上段 代码 在 MyCode\ MySamplel103\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , PathDashPathEffect (myOval, 48, 0，PathDashPathEffect. 
Style. MORPH) 用 于 创建 一 个 路 径 特效 ,myOval 表示 路 径 中 的 图 形 ( 如 椭圆 .矩形 等 ),48 表示 路 径 
中 图 形 与 图 形 之 间 的 间距 ,0 表示 背 个 图 形 的 偏 移 量 ,PathDashPathEffect. Style. MORPH 表示 图 形 
的 摆 放 风格 ,除了 MORPH 风格 之 外 ,还 有 ROTATE 和 TRANSLATE 风格 。myOval. addOval(0， 
0,，46，28，Path. Direction. CCW) 用 于 在 路 径 中 添加 椭圆 图 形 ,(0, 0, 46,，28) 用 于 定义 椭圆 图 形 的 
大 小 ,Path. Direction. CCW 表示 逆 时 针 方向 ,如 果 是 Path. Direction. CW, 则 表示 顺 时 针 方 向 。 此 实 
例 的 完整 项 目 在 MyCode\MySample103 文件 夹 中 ， 


110 ”使 用 SumPathEffect 和 孟 加 多 种 路 径 特效 


此 实例 主要 通过 使 用 SumPathEffect 共 加 两 种 路 径 特 效 , 使 图 像 的 四 周 产生 散射 的 毛刺 。 当 实 
例 运行 之 后 ,图像 的 四 周 产 生 散 射 的 毛刺 效果 如 图 110. 1 所 示 。 


public class MainActivity extends Activity { 
(Override MySample 
protected void onCreate(Bundle savedInstanceState) { 
setContentView(new MYView(this) ); 
super. onCreate( savedInstanceState) ; 
} 
class MyView extends View { 
public MyView(Context context) { super(context); } 
(Override 
protected void onDraw(Canvas myCanvas) { 
Display myDisplay = getWindowManager(). getDefaultDisplay(); 
int myWidth = myDisplay. getWidth( ); 
int myHeight = myDisplay. getHeight(); 
Bitmap myBitmap = 
BitmapFactory. decodeResource ( getResources ( ), R. mipmap. 
myimage); 
PathEffect myEffectl1 = new DiscretePathEffect(5.0f, 70.0f); 
PathEffect myEffect2 = new DiscretePathEffect(1.0f, 50.0f); 
PathEffect myEffect = new SumPathEffect(myEffectl1, myEffect2); 
Paint myPaint = new Paint(); 
myPaint. setPathEffect(myEffect); 
myCanvas. drawOval (myWidth/10, myHeight/20, 
myWidth - myWidth/10, myHeight - myHeight/7, myPaint); // 绘 制 椭圆 
// 以 椭圆 形 裁剪 图 像 
myPaint. setXfermode(new PorterDuffXfermode(PorterDuff. Mode. ADD) ); 
myCanvas. drawBitmap(myBitmap, 0, 30, myPaint); // 绘 制图 像 
}} 
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上 F 面 这 段 代 码 在 MyCode\ MySamplel04\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myEffectl 三 new DiscretePathEffect(5. 0f{，70. 0f) 用 于 
生成 散 列 路 径 特效 ,5. 0f 表示 酚 片 长 度 ,70.0{f 表示 偏 移 量 。myEffect2 = new DiscretePathEffect(1. 
0f，50.0f) 也 用 于 生成 散 列 路 径 特效 ,1. 0f 表示 碎片 长 度 ,50. 0f 表示 偏 移 量 。myEffect 二 new 
SumPathEffect(CmyEffectl ， myEffect2) 用 于 三 加 两 种 路 径 特效 ,myEffectl 表示 第 一 种 路 径 特效 ， 
myEffect2 表示 第 二 种 路 径 特 效 。myPaint， setPathEffect(CmyEffect) 则 表示 在 画笔 中 应 用 二 加 的 路 
径 特效 myEffect。 此 实例 的 完整 项 目 在 MyCode\MySamplel104 文件 夹 中 ，。 


111 通过 BitmapShader 实现 以 图 像 填 充 椭 加 


此 实例 主要 通过 使 用 BitmapShader 类 ,实现 使 用 水 平和 垂直 镜像 图 像 填充 椭圆 内 部 。 当 实例 运 
行 之 后 ,使 用 水 平和 垂直 镜像 图 像 填 充 椭圆 内 部 的 效果 如 图 111. 1 所 示 。 


、 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

(@ Override 

protected void onCreate( Bundle savedInstanceState) { 
setContentView(new MyView(this)); 
super. onCreate(savedInstanceState) ; 

} 

class MyView extends View { 
private BitmapShader myShader = null; 
private Bitmap myBitmap = null; 
private ShapeDrawable myDrawable = null; 
public MyView(Context context) { 

super (context); 
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myBitmap = ((BitmapDrawable) getResources( ) . getDrawable( 


R. mipmap. myimage) ) . getBitmap(); // 获 取 资 源 图 像 
myShader = new BitmapShader(myBitmap, Shader. TileMode.MIRROR, 
Shader. TileMode. MIRROR); // 构 造 泻 染 器 BitmapShader 
} 
protected void onDraw(Canvas myCanvas) { 
Super. onDraw(myCanvas ) ; 
int myWidth = 1080,myHeight = 1820; 
myCanvas. drawColor (Color. BLACK); // 设 置 背 景 颜色 
myDrawable = new ShapeDrawable(new 0valShape( ) ); // 构 建 椭圆 ShapeDrawable 
myDrawable. getPaint(). setShader(myShader); // 获 取 画 笔 并 设置 泻 染 器 
myDrawable. setBounds(20, 20, myWidth - 20, myHeight - 150); // 设 置 显 示 区 域 
myDrawable. draw( myCanvas); // 绘 制 shapeDrawable 


上 和 面 这 上段 代码 在 MyCode\ MySample093\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,BitmapShader (myBitmap, Shader. TileMode. MIRROR 
Shader. TileMode. MIRROR) 用 于 创建 一 个 位 图 演 染 器 BitmapShader ,该 构造 阴 数 的 语法 声明 如 下 : 


public BitmapShader(Bitmap bitmap, Shader. TileMode tilex, Shader. TileMode tileY) 


其 中 ,参数 Bitmap bitmap 表示 在 泻 染 器 内 使 用 的 位 图 。 参 数 Shader. TileMode tileX 表示 位 图 
在 工 方 向 的 堆放 模式 。 参 数 Shader. TileMode tileY 表示 位 图 在 > 方 回 的 堆放 模式 。Shader. 
TileMode 堆放 模式 有 下 列 3 种 : CLAMP、REPEAT、MIRROR.。 

此 实例 的 完整 项 目 在 MyCode\MySample093 文件 夹 中 。 


112 使 用 ComposeShader 创建 渐变 图像 


此 实例 主要 通过 使 用 ComposeShader 二 加 BitmapShader 和 LinearGradient 两 种 着 色 需 ,实现 以 
线性 渐变 透明 的 效果 显示 图 像 。 当 实例 运行 之 后 , 单 击 " 绿 色 渐 变 ?” 按 钮 , 则 在 图 像 的 上 面 蒙 上 一 层 从 
透明 至 绿色 渐变 的 遮 旱 层 , 如 图 112. 1 的 左 图 所 示 ; 单 击 “ 白 色 渐 变 ” 按 钮 , 则 在 图 像 的 上 面 绽 上 一 层 
从 透明 至 白色 渐变 的 遮 音 层 , 如 图 112. 1 的 右 图 所 示 。 单 击 其 他 按钮 将 会 实现 按钮 标题 所 示 的 效果 ， 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

MyView myView; 

Bitmap myBitmap; 

Shader myBmpShader; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myView = (MyView) findViewById(R. id. myView); 
Display myDisplay = getWindowManager( ). getDefaultDisplay(); 
myView. myWidth = myDisplay. getWidth( ); 
myView. myHeight = myDisplay.getHeight(); 
myBitmap = ((BitmapDrawable) getResources( ) . getDrawable( 


R. mipmap. myimage) ) . getBitmap( ); // 获 取 图 像 资 源 
myBmpShader = new BitmapShader(myBitmap, 
Shader. TileMode. REPEAT, Shader. TileMode. REPEAT); // 创 建 BitmapShader 
myView. myShader = myBmpShader; 
} 
public void onClickBtnGreen(View v) { // 响 应 单 击 "绿色 渐变 "按钮 


// 创 建 沿 着 透明 色 至 绿色 的 方向 进行 线性 渐变 的 LinearGradient 

Shader myLinearGradientShader = new LinearGradient(0, 0, 0, myView.myHeight, 
new int[ ] {Color. TRANSPARENT, Color. GREEN}, null, Shader. TileMode. REPEAT); 

myView.myShader = new ComposeShader(myBmpShader, myLinearGradientShader ， 


PorterDuff. Mode. RDD); // 混合 泻 染 , 将 线性 透明 渐变 和 图 像 琶 加 
myView. postInvalidate( ); 
} 
public void onClickBtnWhite(View v) { // 响 应 单 击 "白色 渐变 "按钮 


// 创 建 沿 着 透明 色 至 白色 的 方向 进行 线性 渐变 的 LinearGradient 
Shader myLinearGradientShader = new LinearGradient(0, 0, 0, myView.myHeight, 
new int[ ] {Color. TRANSPARENT, Color.WHITE}, null, Shader. TileMode. REPEAT); 
myView.myShader = new ComposeShader(myBmpShader, 
myLinearGradientShader ,PorterDuff. Mode. ADD); 
myView. postInvalidatel( ); 
} 
public void onClickBtnBlue(View v) { // 响 应 单 击 " 蓝 色 渐变 "按钮 
// 创 建 沿 着 透明 色 至 蓝 色 的 方向 进行 线性 渐变 的 LinearGradient 
Shader myLinearGradientShader = new LinearGradient(0, 0, 0, myView.myHeight, 
new int[ ] {Color. TRANSPARENT, Color. BLUE}, null, Shader. TileMode. REPEAT); 
myView.myShader = new ComposeShader(myBmpShader., 
myLinearGradientShader , PorterDuff. Mode. ADD); 
myView. postInvalidate( ); 
4 


上 面 这 段 代 码 在 MyCode\ MySample097\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myBmpShader 二 new BitmapShader(myBitmap ，Shader. 
TileMode. REPEAT,Shader. TileMode. REPEAT) 用 于 根据 图 像 资 源 创 建 BitmapShader 图 像 着 色 
名 ,myBitmap 表示 图 像 ,Shader. TileMode. REPEAT( 前 一 个 ) 表 示 图 像 在 水 平方 品 上 的 平 铺 模式 ， 
Shader. TileMode.REPEAT( 后 一 个 ) 表 示 图 像 在 垂直 方 回 上 的 平 铺 模 式 。myLinearGradientShader = 
new LinearGradient(0, 0, 0, myView. myHeight, new int| | {Color. TRANSPARENT ,Color. 
BLUE}，null,Shader. TileMode. REPEAT) 用 于 创建 蓝 色 渐 变 透明 的 者 色 右 ,(0, 0) 表 示 渐 变 开 始 位 
置 的 坐标 , (0，myView. myHeight) 表示 渐变 结束 位 置 的 坐标 , {Color. TRANSPARENT, Color. 
BLUE} 表 示 渐 变 的 开始 颜色 是 透明 色 、 结 束 颜色 是 蓝 色 ,null 表示 渐变 过 程 均匀 分 布 , Shader. 
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TileMode. REPEAT 表示 如 果 渐 变 不 能 填充 目标 对 象 , 则 用 重复 模式 平 铺 渐变 效果 。myView-. 
myShader 一 new ComposeShader (myBmpShader, myLinearGradientShader , PorterDutff. Mode. 
ADD) 表 示 使 用 myBmpShader 图 像 着 色 央 和 myLinearGradientShader 线性 渐变 着 色 器 以 
PorterDuff. Mode. ADD 芭 加 的 模式 创建 一 个 组 合 着 色 器 。myView. postInvalidate( ) 用 于 刷新 
myView。myView 是 自 定义 (控件 ) 类 MyView 的 实例 ,MyView 类 的 主要 代码 如 下 : 


public class MyView extends View { 
public Shader myShader = null; 
public Paint myPaint = null; 
public int myWidth, myHeight; 
public MyView(Context context, AttributeSet attrs) { 
super(context, attrs); 
myPaint = new Paint(); 
} 
(Override 
protected void onDraw(Canvas myCanvas) { 
myPaint. setShader(myShader ) ; 
myCanvas. drawRect(0,0,myWidth, myHeight, myPaint); 
大 ， 


上 面 这 上段 代码 在 MyCode\ MySample097\app\src\ main\java\com\bin\luo\mysample\ 
MyView. java 文件 中 。myPaint. setShader(myShader) 用 于 在 画笔 中 设置 包括 图 像 着 色 闫 和 渐变 透 
明 着 色 咒 的 组 合 着 色 嚣 。myCanvas. drawRect(0, 0, myWidth，myHeight，myPaint) 表 示 使 用 组 合 
者 色 器 绘制 窍 形 图 像 。 此 实例 的 完整 项 目 在 MyCode\MySample097 文件 夹 中 。 


113 使 用 ImageView 显示 XML 实现 的 矢量 图 形 


此 实例 主要 通过 设置 ImageView 控件 的 src 属性 为 XML 格式 的 矢量 图 形 文 件 , 实 现在 
ImageView 控件 中 显示 以 矢量 形式 创建 的 桃 心 图 形 。 当 实例 运行 之 后 ,矢量 桃 心 图 形 的 显示 效果 如 
图 113. 1 所 示 ，。 
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主要 代码 如 下 : 


< ImageView android:layout width= "wrap_content" 
android:layout height = "wrap_content" 
android: src = " (Odrawable/myvectordata" 
android: layout centerInParent = "true"/> 


上 面 这 段 代 码 在 MyCode\MySample238\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android: src 王 "人 (drawable/myvectordata "的 myvectordata 是 一 个 XML 格式 的 文件 ， 
该 文件 包含 了 创建 矢量 桃 心 图 形 的 数据 。myvectordata. xml 文件 的 主要 内 容 如 下 : 


< Vector xmlns:android = "http://schemas. android. com/apk/res/android" 

android: width = "480dp" 

android: height = "480dp" 

android: viewportWidth = "480.0" 

android: viewportHeight = "480.0"> 
< path android: strokeColor = " # FF0000" 

android: strokeWidth = "1" 

android:fillColor = " # FF0000" 

android: pathData = " M233. 642 288 C249.9 265.728 318 231. 266 318 195.613 C318 133.896 247. 069 
149. 581 232.993 172.957 C218.041 147.832 149 137.805 149 195.0613 C149 232.135 219. 429 2065.911 233.0642 
288 z" /> 
</vector > 


上 面 这 段 代 码 在 MyCode\MySample238\app\src\main\res\drawable\myvectordata. xml 文件 
中 。 在 这 段 代 码 中 ,矢量 桃 心 图 形 数 据 主要 在 < path > 标签 中 定义 ,线条 粗细 、 颜 色 等 属性 也 在 此 设 
置 。 此 实例 的 完整 项 目 在 MyCode\MySample238 文件 夹 中 。 


114 使 用 BitmapFactory 压缩 图 像 的 大 小 


此 实例 主要 通过 在 BitmapFactory 类 的 decodeFile() 方 法 中 设置 RGB 565 参数 属性 ,实现 使 用 
RGB _565 法 压缩 图 像 大 小 。 当 实例 运行 之 后 ,在 "图像 文件 : "输入 框 中 输入 图 像 文件 的 路 径 , 单 击 
“显示 压缩 前 图 像 ” 按 钮 , 则 在 弹出 的 Toast 中 显示 压缩 前 的 图 像 大 小 和 宽 高 ,如 图 114. 1 的 左 图 所 
示 。 单 击 “ 显 示 压 缩 后 图 像 ” 按 钮 , 则 在 弹出 的 Toast 中 显示 压缩 后 的 图 像 大 小 和 宽 高 ,如 图 114.1 的 
右 图 所 示 。 很 明显 ,压缩 前 后 的 图 像 质量 基本 不 变 , 但 是 大 小 几乎 减 小 一 半 。 需 要 注意 的 是 ,在 测试 
时 ,请 确保 在 SD 卡 上 有 myimg. png 图 像 文件 。 

主要 代码 如 下 : 


public void onClickmyBtnl (View v) { // 响 应 单 击 " 显 示 压 缩 前 图 像 "按钮 
BitmapFactory. Options myOptions = new BitmapFactory. Options( ); 
myOptions. inPreferredConfig = Bitmap. Config.ARGB 8888; 
Bitmap myBmp = 
BitmapFactory. decodeFile(myFile.getText().toString(), myOptions); 
myImage. setInageBitmap(myBmp); 
String myInfo = "压缩 前 的 图 像 大 小 " + (myBmp. getByteCount() / 1024 / 1024) 
+ "M, \n 宽度 为 ”+ myBmp. getWidth() + ",\n 高 度 为 ”+ myBmp. getHeight(); 
Toast. makeText (MainActivity. this, myInfo, Toast. LENGTH_SHORT). show( ); 
} 
public void onClickmyBtn2(View v) { // 响 应 单 击 "显示 压缩 后 图 像 "按钮 


~、 
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BitmapFactory. Options myOptions = new BitmapFactory. Options(); 
myOptions. inPreferredConfig = Bitmap.Config.RGB 565; 
Bitmap myBmp = 

BitmapFactory. decodeFile(myFile.getText().toString(), myOptions); 
myImage. setInageBitmap(myBmp); 
String myInfo = "压缩 后 的 图 像 大 小 " + (myBmp. getByteCount() / 1024 / 1024) 

+ "M, \n 宽度 为 ”+ myBmp. getWidth() + ",\n 高 度 为 "+ myBmp. getHeight(); 

Toast. makeText (MainActivity. this, myInfo, Toast. LENGTH SHORT). show( ) ; 
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图 114. 1 


上 F 面 这 段 代 码 在 MyCode\ MySample234\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myBmp 三 BitmapFactory. decodeFile (myFile. getText ( ). 
toString()，myOptions) 用 于 将 指定 路 径 的 文件 根据 指定 的 参数 myOptions 解码 为 一 个 位 图 对 象 ,在 
参数 myOptions 的 inPreferredConfig 属性 中 ,如果 设置 该 属性 值 为 Bitmap. Config. ARGB _ 8888 , 则 
将 保持 原始 图 像 , 如果 设置 该 属性 值 为 Bitmap. Config. RGB 565, 则 按照 RGB 565 压缩 图 像 。 此 
外 ,在 SD 卡 读 写 文件 需要 在 AndroidManifest. xml 文件 中 添加 权限 < uses 一 permission android: 
name 一 "androld. permission. READ_EXTERNAL_ STORAGE"/> 和 < uses— permission android: 
name 一 "android. permission. WRITE EXTERNAL STORAGE"/>。 此 实例 的 完整 项 目 在 MyCode 
\MySample234 文件 夹 中 ， 


115 在 自 定义 类 中 使 用 Movie 显示 动态 图 像 


此 实例 主要 通过 在 自 定 义 类 MyGifView 中 使 用 Movie 类 的 相关 方法 ,实现 显示 GIF 动态 图 像 。 
当 实 例 运行 之 后 , 单 击 “ 开 始 播放 ”按钮 , 则 显示 GIF 动态 图 像 ; 单 击 “ 停 止 播放 ”按钮 , 则 显示 GIF 静 
态 图 像 , 效 果 分 别 如 图 115. 1 的 左 图 和 右 图 所 示 。 
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MySample 


开始 播放 


图 115. 1 
public void onClickmyBtnl(View v) { // 响 应 单 击 " 开 始 播放 "按钮 
myImageView. setVisibility(View. GONE); 
myGIFImage. setVisibility(View. VISIBLE); 
} 
public void onClickmyBtn2(View v) { // 响 应 单 击 " 停 止 播放 "按钮 


myImageView. setVisibility(View. VISIBLE); 
myGIFImage. setVisibility(View. GONE); 
} 


上 面 这 段 代 码 在 MyCode\ MySample687\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImageView. setVisibility(View. VISIBLE) 用 于 显示 
GIF 静态 图 像 , 该 静态 图 像 通 常 是 GIF 动态 图 像 的 首 幅 图 像 。myGIFImage. setVisibility (View. 
GONE) 用 于 隐藏 静态 图 像 。myGIFImage 是 自 定 义 类 MyGifView 的 实例 , 自 定义 类 MyGifView 的 
主要 代码 如 下 : 


public class MyGifView extends ImageView { 

private Movie myGif; 

private long myGifStartTime; 

private int myWidth; 

private int myHeight; 

public MyGifView(Context context) { super(context); } 

public MyGifView(Context context, AttributeSet attrs) { 
super(context, attrs); 
loadGifImage(R. raw. myimagel ) ; 

} 

Private void loadGifImage( int resourceId) { 
InputStream myInputStream = getResources( ). openRawResource(resourceId) ; 
myGif = Movie. decodeStream(myInputStream) ; 
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if (myGif != null) { 
Bitmap myBitmap = BitmapFactory. decodeStream(myInputStream); 
myWidth = myBitmap. getWidth( ); 
myHeight = myBitmap. getHeight(); 
myBitmap. recycle( ); 
}} 
(Override 
Protected void onDraw(Canvas canvas) { 
ShowGifImage(canvas ) ; 
invalidate( ) ; 
} 
(Override 
protected void onMeasurel( int widthMeasureSpec, int heightMeasureSpec) { 
super. onMeasure(widthMeasureSpec, heightMeasureSpec); 
} 
private boolean showGifImage(Canvas canvas) { 
long now = SystemClock.uptimeMillis(); 
if (myGifStartTime == 0) { myGifStartTime = now; } 
int myDuration = myGif. duration( ); 
if (myDuration == 0) { myDuration = 100; } 
int myRelativeTime = (int) ((now - myGifStartTime) % myDuration); 
myGif. setTime(myRelativeTinme); 
myGif.draw(canvas, (getWidth() - myWidth) / 2, (getHeight() - myHeight) / 2); 
if ((now -~ myGifStartTime) > = myDuration) { 
myGifStartTime = 0; 
return true; 
} 
return false; 
站 


上 面 这 段 代 码 在 MyCode\ MySample687\app\src\ main\java\com\bin\luo\ mysample\ 
MyGifView. java 文件 中 。 需 要 说 明 的 是 ,如 果 在 编译 安装 时 出 现 错 误 , 问 题 可 能 出 在 人 硬件 加 速 上 , 因 
此 需要 在 AndroidManifest. xml 文件 中 关闭 硬件 加 速 , 即 设置 < activity android: name 一 ". 
MainActivity” android: hardwareAccelerated 一" false" >。 此 实例 的 完整 项 目 在 MyCodeN\ 
MySample687 文件 夹 中 。 


116 ”通过 使 用 图 像 作 为 画布 创建 市 水 印 图 像 


此 实例 主要 通过 使 用 Bitmap 对 象 创 建 画 布 , 实 现在 图 像 上 添加 水 印 文字 。 当 实例 运行 之 后 , 显 
示 的 图 像 没 有 水 印 文 字 ,如 图 116. 1 的 左 图 所 示 。 在 "水 印 文字 : ”输入 框 中 输入 内 容 , 如 "纽约 之 夜 ”， 
单 击 “ 添 加 水 印 文字 ”按钮 , 则 在 图 像 上 显示 添加 的 文字 ,如 图 116. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickBtn1 (View v) { // 响 应 单 击 "添加 水 印 文字 "按钮 
Bitmap myOldBmp = ( (BitmapDrawable) myImageView. getDrawable( )).getBitmap( ); 
Bitmap myNewBmp = drawTextToCenter(this, 
myOldBmp, myEditText.getText().toString(),240,Color. YELLOW); 
myImageView. setImageBitmap(myNewBmp); 


public static Bitmap drawTextToCenter(Context context， 
Bitmap bitmap, String text, int size, int color) { 
Paint myPaint = new Paint(Paint.ANTI ALIAS FLAG); 
myPaint. setColor(color); 
myPaint. setTextSizel( size); 
Rect myRect = new Rect(); 
myPaint.getTextBounds(text, 0, text. length(), myRect); 
return drawTextToBitmap( context, bitmap, text, myPaint, myRect, 
(bitmap. getWidth() - myRect.width()) /2, 
(bitmap. getHeight() + myRect.height()) / 2); 
} 
private static Bitmap drawTextToBitmap(Context context, Bitmap bitmap, 
String text, Paint paint, Rect bounds, int paddingLeft, int paddingTop) { 
Bitmap. Config myConfig = bitmap.getConfig( ); 
paint. setDither (true); 
paint. setFilterBitmap(true); 
bitmap = bitmap. copy(myConfig, true); 
Canvas canvas = new Canvas(bitmap); 


paint. setStyle(Paint. Style. STROKE); // 创 建 空心 文字 
paint. setStrokeWidth(1); // 设 置 文字 线条 宽度 


canvas. drawText (text, paddingLeft, paddingTop, paint); 
return bitmap; 


MySample MySample 


水 印 文字 : 纽约 之 夜 水 印 文字 : 知 约 之 夜 
添加 水 印 文字 添加 水 印 文字 
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上 面 这 段 代 码 在 MyCode\ MySample457\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,drawTextToCenter() 方 法 用 于 在 图 像 中 央 绘 制 水 印 文 
字 。drawTextToBitmap() 方 法 用 于 在 图 像 的 指定 位 置 绘 制 水 印 文字 。canvas 一 new Canvas 
(bitmap) 表 示 使 用 Bitmap 创建 Canvas。paint. setStyle(Paint. Style. STROKE) 用 于 创建 空心 风格 


的 Paint。 此 实例 的 完整 项 目 在 MyCode\MySample457 文件 夹 中 。 
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117 通过 操作 根 布 局 实现 将 屏 荞 内 容 保 存 为 图 像 


此 实例 主要 通过 使 用 View 的 draw() 方 法 ,实现 将 屏幕 内 容 保 存 为 一 幅 图 像 。 当 实例 运行 之 后 ， 
单 击 “将 屏幕 内 容 保存 为 一 幅 图 像 > 按 钮 , 则 将 把 当前 屏幕 内 容 ( 通 知 栏 除 外 ) 作 为 一 幅 图 像 保 存在 SD 
卡 的 根 目 录 下 ,如 图 117. 1 的 左 图 所 示 。 在 SD 卡 根 目录 下 的 截屏 图 像 文 件 名 是 “ScreenImage. png”， 
如 图 117. 1 的 右 图 所 示 。 


记 luobin.png 
大 小 1.94 MB 


本 luobinImage.png 
大 小 1.94 MB 


怕 myimg.png 
大 小 2.46 MB 


订 myimg11.png 
大 小 1.54 MB 


screen.png 
大 小 64.74 KB 


Screenimage.png 
大 小 1.94 MB 


a _0ServerSendToService.txt 


全 ” 大 小 0.19 KB 
pe 
l hos _1ServiceAckToServer.txt 
、 J 


” 大 小 0.13 KB 


成 功 将 屏幕 内 容 保存 为 一 幅 图 像 ， 请 _2XgSdkReceiveFromXGService .txt 
查看 在 存储 卡 上 的 ScreeniImage.png 文 件 大 小 0.13 KB 


~ 、 


四 


主要 代码 如 下 : 
public void viewSaveToImage(View view) { // 将 图 像 保 存 为 文件 
Bitmap myBmp = loadBitmapFromView(view); // 把 一 个 View 转换 成 图 像 
FileOutputStream myFileStreanm; 
try { 
boolean ismySDCard = Environment. getExternalStorageState().equals 
(android. os. Environment. MEDIA MOUNTED ) ; // 判 断 手 机 是 否 已 经 安装 了 SD 卡 
if (ismySDCard) { 
File myRoot = 


Environment. getExternalStorageDirectory(); // 获 取 SD 卡 根 目 录 
File myFile = new File(myRoot, "ScreenImage. png" ); 
myFileStream = new FileOutputStream(myFile); 
} else throw new Exception(" 创建 文件 失败 !" ); 
myBmp. compress(Bitmap. CompressFormat. PNG, 90, myFileStreanm); 
myFileStream. flush( ); 
myFileStream. close( ); 
} catch (Exception e) { e.printStackTrace( ); } 
view. destroyDrawingCache( ); 


} 
private Bitmap loadBitmapFromView(View myView) { // 将 View 转换 成 Bitmap 


int myWidth = myView.getWidth( ); 
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int myHeight = myView. getHeight(); 


Bitmap myBmp = Bitmap. createBitmap(myWidth, myHeight, Bitmap.Config.ARGB 8888); 
Canvas myCanvas = new Canvas(myBmp); 

myCanvas. drawColor( Color. WHITE); 

myView. layout(0, 0, myWidth, myHeight ); 


myView. draw(myCanvas ) ; 
return myBmp; 
} 
public void onClickmyBtnl (View v) { // 啊 应 单 击 "将 屏幕 内 容 保 存 为 一 幅 图 像 "按钮 


View myView = myLayout. getRootView( ); 

viewSaveToImage( myView); 

Toast. makeText ( getApplicationContext ( ), "成 功 将 屏幕 内 容 保 存 为 一 幅 图 像 , 请 查看 在 存储 卡 上 的 
ScreenImage. png 文件 " ,Toast.LENGTH SHORT). show( ) ; 
} 


上 面 这 段 代 码 在 MyCode\ MySample459\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 , myView = 二 myLayout. getRootView() 用 于 取得 
RelativeLayout 布局 控件 的 View 对 象 。myView. draw (myCanvas) 用 于 将 RelativeLayout 布局 控件 
上 的 所 有 内 容 绘 制 到 myCanvas 中 ,这 里 实际 上 是 图 像 ; 当 RelativeLayout 充满 整个 屏幕 的 时 候 , 就 
相当 于 除 通 知 栏 之 外 的 截屏 。 此 外 ,在 SD 卡 上 写 和 文件 需要 在 AndroidManifest. xml 文件 中 添加 < 
uses 一 permission android: name 王 "androld. permission. WRITE_ EXTERNAL _ STORAGE"/> 权 
限 。 此 实例 的 完整 项 目 在 MyCode\MySample459 文件 夹 中 。 


118 通过 手 盆 变 化 实现 平移 旋转 缩放 图 像 


此 实例 主要 通过 设置 ImageView 控件 的 android: scaleType= "matrix"” ,然后 在 ImageView 控件 
的 setOnTouchListener() 方 法 中 监听 手势 的 平移 、 放 转 、 缩 放 等 动作 ,并 将 其 应 用 于 Matrix 和 矩阵 , 即 
通过 平移 旋转、 缩放 甜 阵 的 方式 控制 ImageView 控件 的 图 像 实 现 相 应 的 变化 。 当 实例 运行 之 后 ,在 
ImageView 控件 的 图 像 上 使 用 两 个 手指 执行 平移 .旋转 、 缩 放 等 动作 , 即 可 平移 .旋转 、 缩 放 图 像 ,效果 
分 别 如 图 118. 1 的 左 图 和 右 图 所 示 。 


大 中 恒利 


MySample MySample 
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public class MainActivity extends Activity { 
// 初 始 化 起 始 位 置 ,两 指 间 距 , 旋 转角 度 等 参数 
float startX = 0, startY = 0, startDistance = 1f, startRotation = 0; 
PointF middlePoint = new PointF(); 
Matrix myMatrix = new Matrix(), myCurrentMatrix = 
new Matrix(), myPrepareMatrix = new Matrix(); // 初 始 化 矩阵 对 象 

private static final int NONE = 0, DRAG = 1, Z00M = 2; // 手 势 模 式 
Int MODE = NONE; 
int myWidth, myHeight; 
Bitmap myBitmap; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 

super. onCreate( savedInstanceState) ; 

setContentView(R. layout.activity main); 

final ImageView myImageView = (ImageView) findViewById(R. id.myImageView); 

myBitmap = BitmapFactory. decodeResource(getResources(), R.mipmap.myimage); 

DisplayMetrics myMetrics = new DisplayMetrics( ); 

getWindowManager( ) . getDefaultDisplay().getMetrics(myMetrics); 

myWidth = myMetrics.widthPixels; 

myHeight = myMetrics. heightPixels; 

myImageView. setOnTouchListener(new View. OnTouchListener() { 

(@ Override 
public boolean onTouch(View view, MotionEvent motionEvent) { 
if ((motionEvent.getAction()& 
MotionEvent. ACTION MASK) == MotionEvent.ACTION DOWN) { 


MODE = DRAG; // 拖 忠 操 作 , 如 果 取 消 , 则 无 法 平移 图 像 
startX = motionEvent. getX( ) ; 
startY = motionEvent.getY(); // 记 录 手 势 起 始 位 置 


myPrepareMatrix. set(myMatrix); 
} else if ((motionEvent. getRction( ) & 
MotionEvent. ACTION MASK) == MotionEvent.ACTION MOVE) { 
if (MODE == ZOOM) { // 缩 放 操作 
myCurrentMatrix. set(myPrepareMatrix); 
float myRotation = calc Rotation(motionEvent) - startRotation; 
float myDistance = calc Distance(motionEvent); 
float myScale = myDistance / startDistance; 
myCurrentMatrix. postScale(myScale, myScale, 


middlePoint.x, middlePoint.y); // 缩 放 图 像 (和 矩阵 ) 
myCurrentMatrix. postRotate(myRotation, 

middlePoint.x, middlePoint.y); // 旋 转 图 像 (和 矩阵 ) 
myMatrix. set(myCurrentMatrix); 
} else if (MODE == DRAG) { // 执 行 拖 钨 平移 操作 


myCurrentMatrix. set(myPrepareMatrix); 
myCurrentMatrix. postTranslate(motionEvent. getX() - startxXx, 
motionEvent.getY() - startY); 
myMatrix. set(myCurrentMatrix); 
} 
} else if ((motionEvent. getRction( ) & 
MotionEvent. ACTION MASK) == MotionEvent.ACTION UP) { 
MODE = NONE; 
} else if ((motionEvent.getAction()& 
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MotionEvent. ACTION MASK) == MotionEvent.ACTION POINTER DOWN) { 


MODE = ZOOM; 
startDistance = calc Distance(motionEvent); // 重 新 计算 两 手指 间距 离 
startRotation = calc Rotation(motionEvent); // 重 新 计算 旋转 角度 
myPrepareMatrix. set(myImageView. getImageMatrix( ) ); 

calc MiddlePoint(middlePoint, motionEvent); // 重 新 计算 中 心 点 


} else if ((motionEvent.getAction()& 
MotionEvent. ACTION MASK) == MotionEvent.ACTION POINTER UP) { 


MODE = NONE; 


} 

// 根 据 平移 、 旋 转 、 缩 放 之 后 的 矩阵 来 平移 、 旋 转 、 缩 放 图 像 
myImageView. setImageMatrix(myMatrix); 

return true; 


PT 
private float calc_ Distance(MotionEvent event) { 


float x = event. getX(0) — event. getxX(1); 


float y = event.getY(0) — event. getY(1); 
return (float) Math. sqrt(x x* x + y * y); // 计 算 两 个 触 点 间 的 距离 


} 
private void calc MiddlePoint(PointF point, MotionEvent event) { 


float x = event. getX(0) + event. getx(1); 


float y = event.getY(0) + event. getY(1); 
point. set(x / 2, y / 2); // 计 算 旋转 手势 中 心 点 


} 


private float calc Rotation(MotionEvent event) { 
double delta x = (event. getX(0) — event. getX(1) ); 
double delta y = (event. getY(0) — event. getY(1)); 
double radius = Math.atan2(delta y, delta x); 
return (float) Math. toDegrees(radius); // 计 算 旋 转角 度 


} } 


上 和 面 这 段 代 码 在 MyCode\ MySample733\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample733 文件 夹 中 。 


119 使 用 ThumbnailUtils 提取 大 图 像 的 缩 略 图 


此 实例 主要 通过 使 用 ThumbnailUtils 的 extractThumbnail() 方 法 ,实现 从 一 幅 大 图 像 中 提取 对 
应 的 缩 略图 。 当 实例 运行 之 后 , 单 击 “ 显 示 原 图 ”按钮 , 则 将 显示 大 图 像 , 如 图 119. 1 的 左 图 所 示 。 单 
击 “ 显 示 缩 略图 ”按钮 , 则 将 显示 大 图 像 的 缩 略 图 ,如 图 119. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickBtn1 (View v) { // 响 应 单 击 " 显 示 原 图 "按钮 
myImageView. setImageBitmap(myBitmap); 


} 
public void onClickBtn2(View v) { // 响 应 单 击 "显示 缩 略 图 "按钮 


Bitmap smallBmp = ThumbnailUtils.extractThumbnail(this.myBitmap, 150, 150); 
myImageView. setImageBitmap( smallBmp); 
} 


上 面 这 段 代 码 在 MyCode\ MySamplel07\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,smallBmp = 二 ThumbnailUtils. extractThumbnail (this. 
myBitmap，150，150) 用 于 提取 缩 略 图 ,smallBmp 表示 缩 略 图 ,myBitmap 表示 大 图 像 ,150( 前 一 个 ) 
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显示 原 图 显示 缩 略 图 


图 119. 1 


表示 缩 略 图 的 宽度 ,150( 后 一 个 ) 表 示 缩 略图 的 高 度 。 此 实例 的 完整 项 目 在 MyCode\MySamplel107 
文件 夹 中 。 


120 ”通过 采用 取 模 的 方式 实现 轮流 显示 多 幅 图 像 
此 实例 主要 通过 对 数组 索引 取 模 ,实现 轮流 显示 多 幅 图 像 。 当 实例 运行 之 后 ,默认 将 显示 第 一 幅 


图 像 ,如 图 120. 1 的 左 图 所 示 。 单 击 图 像 , 则 将 显示 下 一 幅 图 像 。 图 120. 1 右 图 所 示 的 图 像 是 第 四 幅 
图 像 , 它 是 图 库 的 最 后 一 ead 单 击 此 幅 图 像 则 将 显 we ld 即 轮 流 显 示 。 


MySample 


深入 理解 Spark 


该 心思 里 与 源码 ? 钙 


DUAL 


图 120.1 
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public class MainActivity extends Activity { 
int[ ] myImages = new int[ ]{ R.mipmap.cpp, R.mipmap. java, 
R. mipmap. spark，R.mipmap. jee};  // 定 义 图 像 数 组 
int myCurrent = 0; 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 


LinearLayout myLinearLayout = new LinearLayout(this); // 创 建 线性 布局 管理 器 
super. setContentView(myLinearLayout); // 加 载 线性 布局 管理 器 
// 宽 度 和 高 度 匹 配 父 窗 口 


ViewGroup. LayoutParams myLayoutParams = myLinearLayout. getLayoutParanms( ); 
myLayoutParams.width = LinearLayout.LayoutParams. MATCH PARENT; 
myLayoutParams. height = LinearLayout.LayoutParams.MATCH PARENT; 
myLinearLayout. setLayoutParams(myLayoutParams ) ; 


final ImageView myImageView = new ImageView(this); // 创 建 ImageView 控件 

myImageView. setLayoutParams(myLayoutParanms); // 宽 度 和 高 度 匹配 父 窗口 
myLinearLayout. addView(myImageView); // 将 ImageView 控件 添加 到 LinearLayout 
myImageView. setImageResource(myImages[0]); // 初 始 化 时 显示 第 一 幅 图 像 
myImageView. setScaleType( ImageView. ScaleTYPe. CENTER ) ; // 居 中 显示 图 像 


// 添 加 Click 单 击 事件 响应 显示 下 一 幅 图 像 
myImageView. setOnClickListener(new View. OnClickListener() { 
@ Override 
public void onClick(View v) { // 轮 流 显 示 myImages 数组 的 图 像 
myImageView. setImageResource(myImages[ ++myCurrent % myImages. length] ); 
FF} 


上 面 这 段 代 码 在 MyCode\ MySample002\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImages 数组 中 的 元 素 : R. mipmap. cpp,R. mipmap. 
java, R. mipmap. spark,R. mipmap. jee 等 是 在 添加 图 像 资 源 时 ,Android Studio 自动 添加 的 ,需要 注 
意 , 图 像 文 件 名 一 定 不 要 大 写 。myImageView. setImageResource (myImages[ 十 十 myCurrent % 
myImages. lengthj) 则 以 取 模 的 方式 控制 数组 索引 ,从 而 实现 多 幅 图 像 周 而 复 始 地 轮流 显示 。 例 如 : 
9%4, 余 数 为 1, 因 此 显示 myImagesLlj 图像; 5%4, 余 数 为 1, 也 显示 myImagesLlj」 图 像 ; 即 无 论 
myCurrent 怎样 增加 ,myImages 数组 总 有 对 应 的 图 像 显 示 。 大 多 数 情 况 下 ,使 用 条 件 语句 判断 是 否 
到 达 数 组 的 末端 ,但 是 通过 取 模 ( 取 余 数 ) 运 算 , 则 简化 和 省 略 了 大 量 的 代码 ,并 且 运 行 更 加 高 效 。 此 
实例 的 完整 项 目 在 MyCode\ MySample002 文件 夹 中 。 


121 使 用 ObjectAnimator 创建 上 下 振动 动 团 


此 实例 主要 通过 使 用 PropertyValuesHolder 创建 多 个 平移 和 旋转 等 动画 ,并 使 用 
ObjectAnimator 类 的 ofPropertyValuesHolder() 方 法 组 合 这 些 动画 ,从 而 产生 振动 效果 。 当 实例 运 
行 之 后 , 单 击 “ 开 始 播放 动画 ?按钮 , 则 图 像 ( 电 话机 ) 将 不 停 地 上 下 振动 ,效果 分 别 如 图 121. 1 的 左 图 
和 右 图 所 示 。 


MySample MySample 


开始 播放 动画 停止 播放 动画 开始 播放 动画 停止 播放 动画 


图 121. 1 


public class MainActivity extends Activity { 
ImageView myImageView; 
ObjectAnimator myAnimator; 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
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super. onCreate( savedInstanceState); 
setContentView(R. layout.activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
PropertyValuesHolder myHolderl = 
PropertyValuesHolder. ofFloat("y", 540, 550); 
PropertyValuesHolder myHolder2 = 
PropertyValuesHolder. ofFloat("rotationY", 0, 25); 
PropertyValuesHolder myHolder3 = 
PropertyValuesHolder. ofFloat("rotationxX", 0, 15); 
myAnimator = ObjectAnimator. ofPropertyValuesHolder(myImageView, 
myHolderl1, myHolder2, myHolder3); 
myAnimator. setRepeatCount(ObjectAnimator. INFINITE); 
myAnimator. setRepeatMode( ObjectAnimator. REVERSE); 
myAnimator. setDuration( 90); 
Button myBtnl = (Button) findViewById(R. id. myBtnl ) ; 
// 响 应 单 击 " 开 始 播放 动画 "按钮 
myBtn1. setOnClickListener(new View. OnClickListener() { 
public void onClick(View arg0) { myAnimator. start(); } }); 
Button myBtn2 = (Button) findViewById(R. id.myBtn2); 
// 响 应 单 击 "停止 播放 动画 "按钮 
myBtn2. setOnClickListener( new View. OnClickListener() { 
public void onClick(View arg0) { myAnimator.cancel(); } }); 
上 


上 F 面 这 段 代 码 在 MyCode\ MySamplel73\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myHolderl 三 PropertyValuesHolder. ofFloat ("y"， 
540, 550) 用 于 创建 控件 的 y 坐标 从 540 改变 到 550 的 动画 。myHolder2 = 二 PropertyValuesHolder. 
ofFloat( "rotationY", 0, 25) 用 于 创建 控件 围绕 y 轴 从 0 度 旋转 到 25 度 的 动画 。myHolder3 = 
PropertyValuesHolder. ofFloat(" rotationX", 0, 15) 用 于 创建 控件 围绕 xz 轴 从 0 度 旋转 到 15 度 的 动 
田 。 myAnimator 三 ObjectAnimator. ofPropertyValuesHolder (myImageView，myHolderl ， 
myHolder2 ,myHolder3) 用 于 将 上 述 3 种 动画 添加 到 myImageView 控件 上 ,并 且 3 个 动画 同时 执行 。 
此 实例 的 完整 项 目 在 MyCode\MySample173 文件 夹 中 。 


122 ”使 用 ObjectAnimator 实现 沿 弧 线路 俊平 移 


此 实例 主要 通过 使 用 Path 的 arcTo() 方 法 创建 弧 线 路 径 ,并 指定 该 路 径 为 ObjectAnimator 动画 
的 平移 路 径 ,实现 ImageView 控件 沿 着 弧 线 平移 的 效果 。 当 实例 运行 之 后 , 单 击 “ 开 始 播 放 动 画 ” 按 
钮 , 则 ImageView 控件 (妖怪 图 像 ) 将 沿 着 细 实 线 指示 的 弧 线路 径 平 移 , 效 果 分 别 如 图 122. 1 的 左 图 和 
右 图 所 示 。 

主要 代码 如 下 : 


// 响 应 单 击 " 开 始 播放 动画 "按钮 
public void onClickButton(View v) { 
myImageView. setVisibility(View. VISIBLE); 
Path myPath = new Path( ); 
RectF myRect = new RectF(50, 50, 700, 700); 
myPath. arcTo(myRect, 600, 600); // 创 建 圆 弧 路 径 
ObjectAnimator myAnimator = 
ObjectAnimator. ofFloat(myImageView, View.X, View.Y, myPath); 
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Path myPathInterpolator = new Path(); 

myPathInterpolator. lineTo(0.6f, 0.9f); 

myPathInterpolator. lineTo(0.75f, 0.2f); 

myPathInterpolator. lineTo(1f, 1f); 

myAnimator. setInterpolator(new PathIinterpolator(myPathInterpolator) ); 
myAnimator. setDuration( 5000); 

myAnimator. start( ); 


图 122.1 


上 面 这 段 代 码 在 MyCode\ MySample566\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ，myAnimator 三 ObjectAnimator. ofFloat 
(myImageView，View. X，View. Y，myPath) 用 于 为 myImageView 控件 创建 一 个 平移 动画 , 它 使 
myImageView 控件 在 指定 的 时 间 内 在 myPath 指定 的 弧 线 路 径 上 平移 。 此 实例 的 完整 项 目 在 
MyCode\MySample566 文件 夹 中 。 


123 ”使 用 ObjectAnimator 深 动 显示 多 幅 图 像 


此 实例 主要 通过 使 用 Handler 定时 调用 ObjectAnimator 动画 ,实现 轮 播 多 幅 图 像 。 当 实例 运行 
之 后 ,将 从 右 回 左 依次 轮 播 多 幅 图 像 (电影 海报 ) ,效果 分 别 如 图 123. 1 的 左 图 和 右 图 所 示 。 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
int[ ] myImages = {R.mipmap.myimagel, R.nmipmap.myimage2, 
R.mipmap. myimage3，R. mipmap. myimage4，R. mipmap. myimage5 } ; 

Handler myHandler = new Handler( ); 
int myCurrentIndex = 0; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 

super. onCreate( savedInstanceState) ; 

setContentView(R. layout. activity main); 

final HorizontalScrollView myScrollView = 
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(HorizontalScrollView) findViewById(R. id.myScrollView); 

final int myWidth = getWindowManager().getDefaultDisplay().getWidth( ); 
int myHeight = getWindowManager().getDefaultDisplay().getHeight(); 
final LinearLayout myLayout = (LinearLayout) findViewById(R. id.myLayout); 
for (int i = 0; i< myImages. length; i++) { 

ImageView myImageView = new ImageView(MainActivity.this); 

myImageView. setScaleType( ImageView. ScaleType. FIT XY); 

myImageView. setImageResource(myImages[i]); 

myImageView. setLayoutParams( 

new RelativeLayout. LayoutParams(myWidth, myHeight)); 

myLayout. addView(myImageView); 
} 
myHandler. postDelayed(new Runnable() { 

(Override 

public void run() { 
if (myCurrentIndex > = myImages. length) myCurrentIndex = 0; 
ObjectRnimator myObjectAnimator = ObjectAnimator.ofInt(myLayout, 
scrollx" ,myLayout. getScrollx(), myWidth x myCurrentIndex++ ) ; 


myObjectAnimator. setStartDelay(1000); // 设 置 图 像 停滞 显示 时 长 
myObjectAnimator. setDuration(1000); // 设 置 平移 过 渡 动 画 时 长 
myObjectAnimator. start( ); // 启 动 动画 

myHandler. postDelayed(this, 2000); 

} }, 1000); // 使 用 Handler 发 送 消息 


} } 


上 面 这 段 代 码 在 MyCode\MySample841\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文 件 中 。 在 这 段 代 码 中, myObjectAnimator 三 ObjectAnimator. ofInt (myLayout," scrollX"， 
myLayout. getScrollX() ,myWidth * myCurrentIndex 十 十 ) 表 示 将 myLayout 的 scrollX 属性 值 在 指定 的 
时 间 内 (1000ms) 从 myLayout. getScrollX() 改 变 成 myWidth * myCurrentIndex 十 十 , 即 每 次 回 左 移动 一 
个 屏幕 宽度 。ObjectAnimator. ofInt() 方 法 的 语法 声明 如 下 : 
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public static Objectanimator ofInt(Object target， 
String propertyName, int... values) 


其 中 ,参数 Object target 指定 该 动画 要 操作 的 是 哪个 控件 。 参 数 String propertyName 指定 该 动 
男 要 操作 该 控件 的 哪个 属性 。 人 参数 int... values 是 可 变 长 参数 ,与 ValueAnimator 的 可 变 长 参数 的 意 
义 一 样 , 即 指 这 个 属性 值 是 从 什么 改变 成 什么 。 

此 实例 的 完整 项 目 在 MyCode\MySample841 文件 夹 中 ， 


124 使 用 ObjectAnimator 实现 图 形 数 字形 变 
此 实例 主要 通过 在 ObjectAnimator 动画 中 改变 矢量 图 形 数字 的 pathData 属性 值 ,实现 矢量 图 形 


数字 5 在 单 击 之 后 执行 收缩 挤 压 等 动画 效果 后 变 成 6。 当 实例 运行 之 后 ,将 显示 矢量 图 形 数字 5, 如 
图 124. 1 的 左 图 所 示 ; 单 击 屏幕 , 则 执行 收缩 挤 压 等 动画 变 成 6, 如 图 124. 1 的 右 图 所 示 。 


MySample MySample 


图 124. 1 
主要 代码 如 下 : 


public void onClickImageView(View v){ // 响 应 单 击 ImageView 控件 
AnimatedVectorDrawable myDrawable = 
(AnimatedVectorDrawable) myImageView. getDrawable( ); 
if(!bChecked) myDrawable. start( ); 
else myDrawable. stop( ); 
bChecked = !bChecked; 
} 


上 和 面 这 上段 代码 在 MyCode\ MySample626 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myDrawable. start() 表 示 启 动 ImageView 控件 的 动画 ， 
myDrawable. stop() 表 示 停 止 ImageView 控件 的 动画 ,因此 ImageView 控件 的 动画 不 能 直接 为 纯粹 
的 图 像 资 源 , 而 应 是 XML 文件 。 如 android:src= 二 ="(@drawable/ mytransition" ,表示 在 ImageView 控 
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件 上 加 载 的 XML 动画 mytransition。mytransition 动画 的 主要 内 容 如 下 : 


< animated — vector xmlns:android = "http://schemas. android. com/apk/res/android" 
android: drawable = " @drawable/nyvector"> 
< target android: name = " myPathName" 
android: animation = " (animator/myanimated" /> 
</animated — vector > 


上 面 这 段 代 码 在 MyCode\MySample626\app\src\main\res\drawable\mytransition. xml 文件 
中 。 在 这 段 代 码 中 ,android: name 王 "myPathName" 表 示 上 自 定 义 的 路 径 ( 即 矢量 图 形 数 字 5) 名 称 ， 
android:animation 一 "(@ animator/myanimated" 中 的 myanimated 文件 用 于 指定 动画 的 具体 参数 ， 
myanimated 文件 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android"> 
<objectAnimator android: duration = "1000" 

android: propertyName = " pathData" 

android: valueFrom = " M 0.806629834254144, 0.110497237569061 C 0. 502762430939227，0. 
110497237569061 0. 502762430939227, 0. 110497237569061 0. 502762430939227, 0. 110497237569061 C 0. 
397790055248619, 0. 430939226519337 0. 397790055248619, 0. 430939226519337 0. 397790055248619, 0. 
430939226519337 C 0. 535911602209945, 0. 364640883977901 0. 801104972375691, 0. 469613259668508 0. 
801104972375691, 0. 712707182320442 C 0. 773480662983425, 1. 01104972375691 0. 375690607734807,，1. 
0939226519337 0.248618784530387, 0.850828729281768" 

android: valueTo = " M 0. 607734806629834，0. 110497237569061 C 0. 607734806629834，0. 
110497237569061 0. 607734806629834, 0. 110497237569061 0. 607734806629834, 0. 110497237569061 C 0. 
392265193370166, 0. 43646408839779 0. 265193370165746, 0. 50828729281768 0. 25414364640884, 0. 
696132596685083 C 0. 287292817679558, 1. 13017127071823 0. 87292817679558, 1. 06077348066298 0. 
845303867403315, 0. 696132596685083 C 0. 806629834254144, 0. 364640883977901 0. 419889502762431，0. 
353591160220994 0.295580110497238, 0.552486187845304" 

android: valueType = " pathType" /> 
</ set > 


上 面 这 段 代 码 在 MyCode\MySample626\app\src\main\res\animator\myanimated. xml 文件 中 。 
在 这 段 代 码 中 ,android:valueFrom 属性 值 在 此 处 就 是 矢量 图 形 数字 5 的 数据 ,android:valueTo 属性 
值 在 此 处 就 是 矢量 图 形 数 字 6 的 数据 ,因此 当 objectAnimator 动画 开始 执行 时 ,就 直接 从 上 面 两 个 属 
性 值 进行 过 渡 。 此 外 ,如 果 当 前 工程 项 目的 res 目录 中 不 存在 animator 子 目 录 , 应 该 先 创建 此 子 目 
录 ,再 在 该 子 目录 中 添加 myanimated. xml 文件 。 

在 mytransition. xml 文件 中 的 android: drawable 王 "@ drawable/myvector" 的 myvector 表示 使 
用 pathData 描述 的 矢量 图 形 数 字 5 的 文件 ,myvector 文件 的 主要 内 容 如 下 : 


< Vector xmlns:android = "http://schemas. android. com/apk/res/android" 
android:width = "100dp" 
android: height = "100dp" 
android:tint = "#00f" 
android:viewportHeight = "1" 
android:viewportWidth = "1"> 
< group android:translateX="— 0.06" 
android:translateY= "一 0.06"> 
< path android:name = "myPathName" 
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android: pathData = " M 0. 806629834254144, 0. 110497237569061 C 0. 502762430939227， 0. 
110497237569061 0. 502762430939227，0. 110497237569061 0. 502762430939227, 0. 110497237569061 C 0. 
397790055248619, 0. 430939226519337 0. 397790055248619, 0. 430939226519337 0. 397790055248619， 0. 
430939226519337 C 0. 535911602209945, 0. 364640883977901 0. 801104972375691, 0. 469613259668508 0. 
801104972375691, 0. 712707182320442 C 0.773480662983425, 1. 01104972375691 0. 375690607734807,， 1. 
0939226519337 0.248618784530387, 0.850828729281768" 

android: strokeColor = "#0f0" 

android: strokeWidth = "0.08" /> 
</group></vector > 


上 面 这 段 代 码 在 MyCode\MySample626\app\src\main\res\drawable\myvector. xml 文件 中 。 
在 这 段 代 码 中 ,android:pathData 属性 值 表 示 绘 制 撩 量 图 形 数 字 5 的 数据 。android: strokeWidth 表 
示 矢 量 图 的 线条 宽度 。android:strokeColor 表示 矢量 图 的 线条 颜色 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample626 文件 夹 中 。 


125 ”使 用 ObjectAnimator 改变 图 像 的 色相 值 


此 实例 主要 通过 使 用 属性 动画 ObjectAnimator 改变 自 定义 控件 AnimImageView 的 属性 ,实现 
以 动画 的 形式 动态 改变 ImageView 控件 的 图 像 的 R、G、B 通道 的 色相 值 。 当 实例 运行 之 后 , 单 击 “ 开 
始 色 相 旋 转 ” 按 钮 , 则 将 以 动画 的 形式 依次 旋转 图 像 的 RG、B 通道 的 色相 值 , 效 果 分 别 如 图 125. 1 的 
左 图 和 右 图 所 示 。 单 击 “ 暂 停 色 相 旋转 ”按钮 , 则 将 暂停 ImageView 控件 的 图 像 色 相 旋转 动画 。 


MySample MySample 


开始 色相 旋转 暂停 色相 旋转 开始 色相 旋转 暂停 色相 旋转 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
AnimImageView myImageView; 
ObjectAnimator myObjectAnimator; 
(Override 


protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity_main); 
myImageView = (MnimImageView) findViewById(R. id.myImageView); 
// 自 定义 属性 动画 
myObjectAnimator = ObjectAnimator. ofFloat(myImageView," myDegree" ,Of,360f); 
// 设 置 动 画 时 长 和 重复 次 数 , - 1 表示 无 限 执行 
myObjectAnimator. setDuration(5000). setRepeatCount( - 1); 


} 
// 响 应 单 击 " 开 始 色 相 旋 转 "按钮 
public void onClickBtnl (View v) { 
if(myObjectAnimator. isPaused() ){ 


myObjectAnimator. resume( ); // 继 续 动 画 
} else{ 

myObjectAnimator. start( ); // 第 一 次 执行 动画 
4 
// 响 应 单 击 " 暂 停 色 相 旋 转 " 按 钮 


public void onClickBtn2(View v) { 
if(myObjectAnimator. isStarted() ){ 
myObjectAnimator. pause( ); // 暂 停 动画 
1 


上 面 这 段 代 码 在 MyCode\ MySample769\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中, myObjectAnimator 三 ObjectAnimator. ofFloat 
(myImageView,"myDegree" ,0f,360f) 用 于 创建 属性 动画 。ObjectAnimator. ofFloat() 方 法 的 语法 声 


明 如 下 : 


public static ObjectAnimator ofFloat (Object target, 
String propertyName, float... values) 


其 中 ,Object target 参数 用 于 指定 这 个 动画 操作 的 控件 。String propertyName 参数 用 于 指定 这 
个 动画 操作 的 属性 。float... values 参数 是 可 变 长 参数 , 即 指示 属性 值 改 变 范 围 ,在 此 实例 中 , 即 是 将 
myImageView 控件 的 myDegree 属性 从 0f 改变 到 360f。 

myImageView = (AnimImageView)findViewByIdCR. id. myImageView) 的 AnimImageView 是 
在 ImageView 控件 的 基础 上 创建 的 自 定义 控件 ,AnimImageView 类 (控件 ) 的 主要 代码 如 下 : 


public class AnimImageView extends ImageView { 
float myDegree = 0; 
ColorMatrix myMatrix; 
public float getMyDegree() { return myDegree; } 
public void setMyDegree(float degree) { myDegree = degree; } 
public AnimImageView(Context context, AttributeSet attrs) { 
super(context, attrs); 
myMatrix = new ColorMatrix( ); // 初 始 化 ColorMatrix 对 象 
} 
(Override 
Protected void onDraw(Canvas canvas) { 
super. onDraw(canvas ) ; 


myMatrix. setRotate(0, myDegree); // 旋 转 RR 通道 的 色相 值 
myMatrix. setRotate(1, myDegree); // 旋 转 G 通道 的 色相 值 
myMatrix. setRotate(2, myDegree); // 旋 转 B 通道 的 色相 值 
// 在 图 像 中 应 用 ColorMatrix 对 象 


setColorFilter(new ColorMatrixColorFilter(myMatrix) ); 
}} 
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上 F 面 这 段 代 码 在 MyCode\ MySample769\app\src\ main\java\com\bin\luo\mysample\ 
AnimImageView. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample769 文件 夹 中 。 


126 ”使 用 AnimatorSet 组 合 多 个 ObjectAnimator 


此 实例 主要 通过 使 用 AnimatorSet 组 合 多 个 属性 动画 ,实现 图 像 同 时 产生 旋转 、 缩 放 等 多 种 动画 
效果 。 当 实例 运行 之 后 , 单 击 “开始 播放 动画 按钮 , 则 图 像 ( 妖 怪 ) 将 以 自身 为 中 心 进行 旋转 、 缩 小 直 
到 消失 ,在 图 像 缩 小 的 过 程 中 ,图 像 的 透明 度 也 将 逐渐 变 小 ,效果 分 别 如 图 126. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


开始 播放 动画 停止 播放 动画 开始 播放 动画 停止 播放 动画 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
ImageView mylImageView; 
AnimatorSet myAnimatorSet; 
(@ Override 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
// 创 建 透明 度 动画 
ObjectAnimator myObjectAnimatorl1 = 
ObjectAnimator. ofFloat(myImageView, "alpha", 1f, 0f); 
// 创 建 旋转 动画 
ObjectAnimator myObjectAnimator2 = 
ObjectAnimator. ofFloat(myImageView," rotation", 0f, 1800f); 
// 创 建 水 平 缩放 动画 
ObjectAnimator myObjectAnimator3 = 
ObjectAnimator. ofFloat(myImageView, " scaleXx", 1.0f, 0f); 
// 创 建 垂直 缩放 动画 
ObjectAnimator myObjectAnimator4 = 


ObjectAnimator. ofFloat(myImageView, "scaleY", 1.0f, 0f); 
// 创 建 动画 集合 
myAnimatorSet = new AnimatorSet(); 
// 在 动画 集合 中 添加 四 种 动画 
myAnimatorSet. play(myObjectAnimator1).with(myObjectAnimator2). 

with(myObjectAnimator3).with(myObjectAnimator4); 

// 设 置 动画 持续 时 间 9 秒 
myAnimatorSet. setDuration( 9000); 


} 
// 响 应 单 击 "开始 播放 动画 "按钮 
public void onClickBtnl (View v) { myAnimatorSet. start(); } 


// 响 应 单 击 "停止 播放 动画 "按钮 
public void onClickBtn2(View v) { myAnimatorSet. cancel(); } 


} 


上 和 面 这 上段 代码 在 MyCode\ MySamplel61 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myAnimatorSet. play (myObjectAnimatorl ). with 
(myObjectAnimator2 ). with ( myObjectAnimator3 ). with (myObjectAnimator4 ) 用 于 在 
myAnimatorSet 动画 集合 加 载 myObjectAnimatorl、myObjectAnimator2、myObjectAnimator3 和 
myObjectAnimator4 四 种 动画 。AnimationSet 与 AnimatorSet 最 大 的 不 同 在 于 , AnimationSet 使 用 
的 是 Animation 子 类 ,AnimatorSet 使 用 的 是 Animator 的 子 类 。Animation 是 针对 视图 外 观 的 动画 
实现 ,动画 被 应 用 时 外 观 改 变 ,但 视图 的 触发 点 不 会 发 生变 化 , 仍 在 原来 定义 的 位 置 。Animator 是 针 
对 视图 属性 的 动画 实现 ,动画 被 应 用 时 对 象 属性 产生 变化 ,最终 导 致 视图 外 观 变化 。AnimationSet 最 
常用 的 操作 是 调用 其 addAnimation () 方 法 将 多 个 不 同 的 动画 组 织 起 来 ,然后 调用 view 对 象 的 
startAnimation() 方 法 触发 这 些 动画 执行 。AnimatorSet 最 第 用 的 操作 是 调用 其 play()、before()、 
with() after() 等 方法 设置 动画 的 执行 顺序 ,然后 调用 start() 方 法 触发 动画 。 此 实例 的 完整 项 目 在 
MyCode\MySamplel61 文件 夹 中 。 


127 使 用 TypeEvaluator 实现 颜色 过 渡 动 画 


此 实例 主要 通过 使 用 ObjectAnimator 创建 颜色 动画 ,并 在 ofObject() 方 法 中 自 定义 
TypeEvaluator ,实现 控件 的 背景 从 一 种 颜色 自然 过 渡 到 另 一 种 颜色 。 当 实例 运行 之 后 , 单 击 “开始 播 
放 动 画 ” 按 钮 , 则 背景 颜色 将 在 5 秒 内 从 灰色 逐渐 改变 成 绿色 ,效果 分 别 如 图 127. 1 的 左 图 和 右 图 所 
示 。 单 击 * 停 止 播放 动画 ?按钮 , 则 停止 颜色 动画 的 改变 动作 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 

ImageView myImageView; 

ObjectAnimator myObjectAnimator; 

LinearLayout myLayout; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
myLayout = (LinearLayout) findViewById(R. id.myLayout); 
int startColor = OxO0ffffff; 
int endColor = 0xff008000; 
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myObjectAnimator = ObjectAnimator.ofObject(myLayout, 
"backgroundColor", new TypeEvaluator() { // 创 建 背 景 颜色 改变 动画 
(Override 


public Object evaluate(float fraction, Object startValue, Object endValue){ 
int startInt = (Integer) startValue; 


int startA = (startInt >> 24) &0xff; 
int startR = (startInt >> 16) &0xff; 
int startG = (startInt >> 8) &0xff; 
int startB = startInt & Oxff; 


int endInt = (Integer) endValue; 
= (endInt >> 24) &0xff; 
int endR = (endInt >> 16) &0xff; 
= (endInt >> 8) &0xff; 
int endB = endInt & Oxff; 
return (int) ((startA + (int) (fraction * (endR - startA))) << 24) | 
(int) ((startR + (int) (fraction * (endR - startR))) << 16) | 
(int) ((startG + (int) (fraction * (endG - startG))) << 8) | 
(int) ((startB + (int) (fraction * (endB - startB)))); 
} }, startColor, endColor); 
myObjectAnimator. setDuration( 5000); // 设 置 动画 持续 时 间 5 秒 
} 
// 响 应 单 击 " 开 始 播放 动画 "按钮 
public void onClickBtnl (View v) { myObjectAnimator. start(); } 
// 响 应 单 击 " 停 止 播放 动画 "按钮 
public void onClickBtn2(View v) { myObjectAnimator. cancel(); } 


MySample MySample 


开始 播放 动画 停止 播放 动画 开始 播放 动画 停止 播放 动画 


127.1 


上 面 这 段 代 码 在 MyCode\ MySamplel62\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myObjectAnimator 一 ObjectAnimator. ofObject 
(myLayout, "backgroundColor", new TypeEvaluator() { )，startColor，endColor) 用 于 创建 一 个 颜 
色 改 变动 画 , myLayout 表示 动画 执行 对 象 ,backgroundColor 表示 动画 执行 对 象 的 属性 名 称 ， 
startColor 表示 动画 起 始 值 ,endColor 表示 动画 结束 值 。TypeEvaluator() 用 于 上 自 定 义 估 值 器 配置 颜 


第 5 章 a 画 《> 


色 从 动画 起 始 值 以 何 种 方式 改变 成 动画 结束 值 , 在 Android 中 ,如果 使 用 ObjectAnimator 的 ofObject() 方 
法 创建 属性 动画 , 通 稼 需要 定义 此 行为 。 此 实例 的 完整 项 目 在 MyCode\MySample162 文件 夹 中 。 


128 ”通过 trimPathEnd 实现 动态 生成 手指 图 形 
此 实例 主要 通过 在 objectAnimator 动画 中 改变 trimPathEnd 属性 值 从 0 到 1(0% 到 100%), 实 


现 动态 生成 手指 矢量 图 形 的 特效 。 当 实例 运行 之 后 , 单 击 屏幕 , 则 将 从 右 端 局 动手 指 拓 量 图 形 的 动态 
绘制 过 程 ,直到 最 后 完成 ,效果 分 别 如 图 128. 01 的 左 图 和 右 图 所 示 。 


MySample MySample 


图 128.1 
主要 代码 如 下 : 


public void onClickImageView(View v) { // 响 应 单 击 ImageView 控件 
bChecked = !bChecked; 
final int[ ] myStateSet = {android.R.attr. state checked * (bChecked ?1: -1))}); 
myImageView. setImageState(myStateSet, true); 

} 


上 面 这 段 代 码 在 MyCode\ MySample617\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImageView. setImageState(myStateSet，true) 用 于 根 
据 当 前 myStateSet 的 值 , 确定 在 单 击 myImageView 控件 时 是 否 执行 或 终止 动画 ; 因此 
myImageView 的 src 属性 不 能 直接 为 图 像 资 源 ,而 应 是 XML 文件。 主要 代码 如 下 : 


< ImageView android: id= "(@ + id/myImageView" 
android:layout width= "match parent" 
android:layout_ height = "match parent" 
android:layout margin = "50dp" 
android:layout centerInParent = "true" 
android:onClick = "onClickImageView" 
android: src = "(Odrawable/mystage" /> 
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上 面 这 段 代 码 在 MyCode\MySample617\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android:src= 王 "(@ drawable/mystage" 用 于 设置 在 myImageView 控件 上 加 载 的 XML 
动画 mystage。mystage 动画 的 主要 内 容 如 下 : 


<?xml version = "1.0" encoding = "utf 一 8"?> 
< animated - selector xmlns:android = "http://schemas. android. com/apk/res/android"> 
< item android: id = "(@ + id/myStart" 
android: state checked = "true" 
android:drawable = "(@drawable/myvector"/> 
< item android: id= "(@ + id/myFinish" 
android:drawable = "(@drawable/myvector"/> 
<transition android: drawable = " (Odrawable/mytransition" 
android:fromId = "(Wid/myStart" 
android: toId = " @id/myFinish" /> 
</animated — selector > 


上 面 这 段 代 码 在 MyCode\MySample617\app\src\main\res\drawable\mystage. xml 文件 中 。 在 
这 段 代 码 中 ,android: drawable 王 "(@ drawable/myvector" 中 的 myvector 是 矢量 图 形 文件 ,myvector 
文件 的 主要 内 容 如 下 : 


< Vector xmlns:android = "http://schemas. android. com/apk/res/android" 
android:width = "240dp" 
android: height = "340dp" 
android:tint = "#f00" 
android:viewportWidth = "24" 
android:viewportHeight = "34"> 
< path android: name = " myPathName" 
android:pathData= "M22,23 q0,4 -4,4h-7q-2,0 -3,-1T1,16q-0.6,-0.80,-2t5,3ql,12, 
Ona 0 = DO I tI UI 0 a0 0 0 0 50 0 i Iltl ll 
i111 90.25 .0.50.5 .0.5 t0.5.=0,.50.5. -$90.2. -=11 =-1t1i10.5.990.25 .0.50.5,.0.5 0.5. =0.51.2, 
-6.5 q0.3, -11,-1t0.8,1 -0.8,6 1722,23" 
android:trimPathEnd = "0" 
android: strokeColor = " # 000" 
android: strokeWidth = "0.2"/> 
</vector > 


上 面 这 段 代 码 在 MyCode\MySample617\app\src\main\res\drawable\myvector. xml 文件 中 。 
在 这 段 代 码 中 ,android:pathData= 二 "M22,23 q0,4 一 4,4 h 一 7 q 一 2,0 一 3, 一 1] Tl1,16 q 一 0.6, 一 0.8 
0, 一 215,3 ql,1 2,0 T8,4 gq0,—1 0.9,—1.1t1.1,1 1.5,9 q0.25,0.5 0.5,0.5 10.5,—0.5 0,—11 
.Zo™— Lleol ll ll oD oo 0 00 oD 03509 9 0 25 1 1s] thsl 
0.5,9 q0. 25,0.5 0.5,0.5t0.5, 一 0.5 1.2, 一 6.5 q0.3, 一 1 1, 一 1 t0.8,1 一 0.8,6 T22,23" 表 示 绘 制 
手指 矢量 图 的 数据 。android: strokeWidth 王 "0. 2" 表 未 矢量 图 的 线条 宽度 。android: strokeColor 王 " 
# 000" 表 示 矢 量 图 的 线条 颜色 。android:trimPathEnd = 二 "0" 表 示 不 显示 该 撩 量 图 (因此 启动 应 用 时 ， 
屏幕 显示 空白 ) ,android:trimPathEnd 王 "1" 表 示 完 全 显示 该 矢量 图 。android:name 王 “myPathName" 表 
示 该 矢量 图 的 上 月 定义 名 称 , 在 调用 动画 时 需要 此 名 称 。 

在 mystage. xml 文件 中 的 代码 android: drawable= 二 "(@drawable/mytransition" 用 于 指定 日 标 动 
男 | mytransition ,mytransition 动画 的 主要 内 容 如 下 : 
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< animated — vector xmlns:android = "http://schemas. android. com/apk/res/android" 
android:drawable = "(@drawable/myvector"> 
<target android: name = "myPathName" 
android: animation = " (Wanimator/myanimated" /> 
</animated — vector > 


上 面 这 段 代 码 在 MyCode\MySample617\app\src\main\res\drawable\mytransition. xml 文件 
中 。 在 这 段 代 人 码 中 ,android: name 二 "myPathName" 表 示 自 定义 的 路 径 ( 矢 量 图 ) 名 称 , android: 
animation 一 "(Oanimator/myanimated" 中 的 myanimated 文件 用 于 指定 trimPathEnd 属性 动画 的 具体 
参数 ,trimPathEnd 属性 表示 截 掉 从 某 个 位 置 到 终点 的 部 分 ,保留 剩 下 的 部 分 。myanimated 文件 的 主 
要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<set xmlns:android = "http://schemas. android. com/apk/res/android"> 
< objectAnimator android: duration = "5000" 
android: interpolator = " (Qandroid: interpolator/linear" 
android: propertyName = " trimPathEnd" 
android: valueFrom= "0" 
android:valueTo = "1"/> 
</set > 


上 面 这 段 代码 在 MyCode\MySample617\app\src\main\res\animator\myanimated. xml 文件 中 。 
在 这 段 代 人 码 中 , android: propertyName 用 于 指定 该 矢量 图 将 要 发 生 改 变 的 属性 名 ,android: 
valueFrom 用 于 设置 动画 初始 值 ,android:valueTo 用 于 设置 动画 结束 值 ; android:valueFrom 王 "0" 和 
android:valueTo 二 "1" ,表示 矢量 图 从 无 到 有 ; android: valueFrom 王 "1" 和 androlid: valueTo= 王 "0" 表 
示 矢 量 图 从 有 到 无 。 换 句 话 说 ,如 果 objectAnimator 的 androlid:propertyName 王 "trimPathEnd", 则 
android:valueFrom 王 "0" 和 android:valueTo="1" 表 示 路 径 ( 矢 量 图 ) 从 起 点 增长 到 终点 , 即 初 始 截断 
部 分 是 100% ,从 起 点 开始 逐渐 缩小 到 终点 ,达到 0% ; android:valueFrom 王 "1" 和 android:valueTo 二 "0" 
表示 路 径 ( 矢 量 图 ) 从 终点 缩短 到 起 点 , 即 初 始 截 断 部 分 是 0 名 ,从 终点 开始 逐渐 扩大 到 起 点 ,达到 
100%。 此 实例 的 完整 项 目 在 MyCode\MySample617 文件 夹 中 。 


129 ”使 用 ValueAnimator 动态 改变 扇形 转角 


此 实例 主要 通过 在 自 定 义 View 中 使 用 ValueAnimator 动画 控制 扇形 的 转角 变化 ,实现 以 动画 的 


形式 动态 绘制 不 同 角 度 的 扇形 。 当 实例 运行 之 后 ,扇形 的 转角 将 沿 着 顺 时 针 方 向 从 小 变 大 ,再 沿 着 道 
时 针 方 向 从 大 变 小 ,并 且 永 不 停歇 ,效果 分 别 如 图 129. 1 的 左 图 和 右 图 所 示 。 
主要 代码 如 下 : 


< com. bin. luo. mysample. myImageView android: id= "(@ + id/myView" 
android: layout width= "wrap_content" 
android:layout height = "wrap_content" 
android: scaleType = "center" 
android:visibility= "visible"/> 


上 面 这 段 代 码 在 MyCode\MySamplel76\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,com. bin. luo. mysample. myImageView 即 是 用 于 以 动画 的 形式 改变 扇形 转角 大 小 的 
和 月 定义 控件 ,myImageView 类 (控件 ) 的 主要 代码 如 下 : 
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public class myImageView extends View { 
private float myDegrees; 
private Paint myPaint; 
public myImageView(Context context, AttributeSet attrs) { 
super(context, attrs); 
myPaint = new Paint(Paint.ANTI ALIAS FLAG); 
myPaint. setColor( Color. BLUE); 
startAnimation( ); 
} 
(Override 
protected void onDraw(Canvas canvas) { myDrawText(canvas); } 
private void myDrawText(Canvas myCanvas) { 
int myHeight = getHeight(); 
int myWidth = getWidth( ); 
myCanvas. drawArc(new RectF(0,0,myWidth, myHeight), 
0f, myDegrees, true, myPaint); 
} 
private void startAnimation() { 
float startDegrees = 0f; 
float endDegrees = 315f; 
ValueAnimator myValueAnimator = 
ValueAnimator. ofFloat( startDegrees, endDegrees); 
// 监 听 角 度 的 改变 
myValueAnimator. addUpdateListener( 
new ValueAnimator. RnimatorUpdateListener() { 
(Override 
public void onAnimationUpdate(ValueAnimator myAnimator) { 
myDegrees = (float)myAnimator.getAnimatedValue( ); 
invalidate( ); 
} }); 
// 设 置 动 画 重复 模式 为 反 向 重复 
myValueAnimator. setRepeatMode( ValueAnimator. REVERSE); 


// 根 据 变化 的 角度 绘制 扇形 


// 开 始 角度 
// 结 束 角 度 


// 获 取 当 前 角度 大 小 
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// 设 置 动 画 重 复 次 数 为 无 限 次 

myValueAnimator. setRepeatCount(ValueAnimator. INFINITE); 

myValueAnimator. setDuration(5000); // 设 置 动 画 持续 时 间 为 5 秒 
myValueAnimator. start( ); // 启 动 动 画 


} } 


上 F 面 这 段 代 码 在 MyCode\ MySamplel76\app\src\main\java\com\bin\luo\ mysample\ 
myImageView. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySamplel176 文件 夹 中 。 


130 ”使 用 ValueAnimator 实现 分 段 转圈 动画 


此 实例 主要 通过 在 自 定义 View 中 使 用 PathMeasure 的 getSegment() 方 法 对 圆 形 路 径 进行 分 段 
截取 ,实现 显示 分 段 圆 弧 转 圈 的 动画 效果 。 当 实例 运行 之 后 ,分 段 圆 踊 转圈 效果 分 别 如 图 130. 1 的 左 
图 和 右 图 所 示 。 

主要 代码 如 下 : 


< com.bin. luo.mysample. PathView android: layout width= "match parent” 
android:layout_height = "match_parent" /> 


上 面 这 段 代码 在 MyCode\MySample851\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,com. bin. luo. mysample. PathView 即 是 实现 分 段 圆 弧 转 圈 动 画 的 自 定义 控件 ， 
PathView 类 的 主要 代码 如 下 : 


Sl Od | 自 1 ) 


MySample MySample 


图 130. 1 


public class PathView extends View { 
private Path myPath; 
private Paint myPaint; 
private PathMeasure myPathMeasure; 
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private float myAnimatedValue; 
private Path newPath; 
private float myLength; 
public PathView(Context context, AttributeSet attrs) { 
super(context, attrs); 
myPathMeasure = new PathMeasurel( ); 
myPaint = new Paint(Paint.ANTI ALIAS FLAG); 
myPaint. setStyle(Paint. Style. STROKE); 
myPaint. setColor( Color. BLUE); 
myPaint. setStrokeWidth(25); 
myPath = new Path(); 
myPath. addCircle(250, 300, 200, Path.Direction. CW); 
myPathMeasure. setPath(myPath, true); 
myLength = myPathMeasure. getLength( ); 
newPath = new Path(); 
final ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, 1); 
valueAnimator.addUpdateListener(new ValueAnimator. AnimatorUpdateListener( ){ 
(Override 
public void onAnimationUpdate(ValueAnimator valueAnimator) { 
myAnimatedValue = (float) valueAnimator.getAnimatedValue( ); 
invalidate( ); 
} }); 
valueAnimator. setDuration(2000); 
valueAnimator. setRepeatCount(ValueAnimator. INFINITE); 
valueAnimator. start( ); 
} 
(Override 
protected void onDraw(Canvas canvas) { 
super. onDraw( canvas); 
canvas. translate(250, 500); // 平 移 画 布 至 (250, 500) 
newPath. reset( ); 
newPath. lineTo(0, 0); 
float myStop = myLength * myAnimatedValue; 
float myStart = (float) (myStop - ((0.5 - 
Math.abs(myAnimatedValue - 0.5)) * myLength)); 
myPathMeasure. getSegment (myStart, myStop, newPath, true); 
canvas. drawPath(newPath, myPaint); // 绘 制 各 段 圆 驱 (分 割 的 路 径 ) 
古 


上 F 面 这 段 代 码 在 MyCode\ MySample851 \app\src\ main\java\com\bin\luo\mysample\ 
PathView. java 文件 中 。 在 这 段 代 码 中 ,myPathMeasure. getSegment(myStart，myStop ，newPath ， 
true) 用 于 截取 将 要 绘制 圆 弧 的 位 置 和 大 小 。getSegment() 方 法 的 语法 格式 如 下 : 


public boolean getSegment (float startD, 
float stopD, Path dst, boolean startWithMoveTo) 


其 中 ,参数 float startD 表示 该 (截取 的 圆 孤 ) 线 段 的 起 点 距离 原始 Path 起 点 的 长 度 。 参 数 float 
stopD 表示 该 (截取 的 圆 孤 ) 线 段 的 终点 距离 原始 Path 起 点 的 长 度 ,参数 Pathdst 表示 截取 的 圆 弧 线 
段 将 添加 到 dst 中 ,注意 是 添加 ,而 不 是 蔡 换 。3 个 参数 的 关系 是 : 0 过 startD 一 stopD 近 Path。 人 参数 
boolean startWithMoveTo 表示 起 始点 是 否 使 用 moveTo 用 于 保证 截取 的 Path 第 一 个 点 位 置 不 变 。 

getSegment() 方 法 的 返回 值 用 于 判断 截取 是 否 成 功 , 如 果 返 回 值 为 true 表示 截取 成 功 , 结 果 存 入 
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dst 中 ; 如 果 返 回 值 为 false 表示 截取 失败 ,不 会 改变 dst 中 内 容 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample851 文件 夹 中 。 


131 使 用 YalueAnimator 在 三 维 Z 轴 上 平移 图 像 


此 实例 主要 通过 使 用 ValueAnimator 的 ofFloat() 方 法 设置 > 轴 和 平移 参数 的 变化 范围 ,再 通过 
ValueAnimator 的 getAnimatedValue() 方 法 获取 适时 z 轴 平 移 参 数值 ,实现 图 像 在 x 轴 上 进行 平移 。 
当 实 例 运 行 之 后 , 单 击 “开始 播放 动画 ”按钮 , 则 图 像 将 在 > 轴 上 进行 平移 ,由 于 < 轴 垂 直 于 屏幕 ,因此 
平移 图 像 与 缩放 图 像 的 效果 相当 类 似 ,效果 分 别 如 图 131. 1 的 左 图 和 右 图 所 示 。 
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开始 播放 动画 


MEMORIAL DAY 


, 人 : 


图 131. 1 
主要 代码 如 下 : 
public void onClickButton(View v) { // 响 应 单 击 "开始 播放 动画 "按钮 
ValueAnimator myValueAnimator = 
ValueAnimator. ofFloat(0, 900); //0,900 表示 图 像 在 z 轴 的 平移 范围 
myValueAnimator. setDuration( 3000); // 设 置 动画 持续 时 间 3 秒 
// 设 置 重 复 次 数 为 无 限 次 


myValueAnimator. setRepeatCount(ValueAnimator. INFINITE); 
myValueAnimator. setRepeatMode(ValueAnimator. REVERSE); // 设 置 重 复 模式 为 反 向 
myValueAnimator. addUpdateListener(new ValueAnimator. AnimatorUpdateListener( ){ 
@ Override 
public void onAnimationUpdate(ValueAnimator animation) { 
float myValue = (float) animation.getAnimatedValue( ); 
Bitmap myNewBmp = GetNewImage(myBitmap, myValue);  // 根 据 改变 的 平移 值 重 绘图 像 


myImageView. setImageBitmap(myNewBmp); // 显 示 新 图 像 
} })); 
myValueAnimator. start( ); 


} 
public Bitmap GetNewImage(Bitmap myOldBmp, float myValue) { 
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BitmapDrawable myDrawable = 
(BitmapDrawable) getResources().getDrawable(R.mipmap. myimage); 
Bitmap oldBmp = myDrawable. getBitmap( ); 
myCamera. save( ); // 保 存 状 态 
Matrix myMatrix = new Matrix(); 
myCamera. translate(0, 0, myValue); 
myCamera. getMatrix(myMatrix); 
myCamera. restore( ); // 恢 复 状 态 
// 设 置 图 像 处 理 的 中 心 点 
myMatrix. preTranslate(oldBmp. getWidth() >> 1, oldBmp. getHeight() >> 1); 
Bitmap newBmp = Bitmap. createBitmap(oldBmp, 0, 0, oldBmp. getWidth( ), 
oldBmp. getHeight(), myMatrix, true); // 通 过 矩阵 生成 新 图 像 


return newBmp; 


上 面 这 段 代 码 在 MyCode\ MySamplel80\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myCamera. translate(0, 0, myValue) 用 于 在 x、y、z 三 个 
轴 上 平移 图 像 , 由 于 xz、y 轴 的 值 均 为 0, 是 动画 只 向 xz 轴 传 值 ,因此 只 有 x 轴 的 平移 运动 。 此 实例 的 完 
整 项 目 在 MyCode\MySamplel80 文件 夹 中 ， 


132 ”使 用 ValueAnimator 实现 起 飞 转 平 飞 动画 


此 实例 主要 通过 在 自 定 义 TypeEvaluator 的 evaluate() 方 法 中 监听 运动 路 径 的 y 坐标 ,并 使 用 
ValueAnimator 操作 该 自 定 义 TypeEvaluator, 实 现在 飞机 图 像 开 始 ( 起 飞 ) 时 以 仰角 30 度 姿 势 运 动 
(飞行 ) ,在 到 达 指 定位 置 (y 二 300) 时 ,以 水 平 姿势 运动 (飞行 ) 的 动态 变换 姿势 效果 。 当 实例 运行 之 
后 , 单 击 “启动 动画 ?按钮 , 则 飞机 图 像 在 开始 (起 飞 ) 时 以 仰角 30 度 姿势 运动 (飞行 ) ,如 图 132. 1 的 左 
图 所 示 ; 在 到 达 指 定位 置 (y==300, 细 线 为 飞行 轨迹 ) 时 ,以 水 平 姿势 运动 (飞行 ) ,如 图 132. 1 的 右 图 
所 示 。 
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图 132.1 


主要 代码 如 下 : 


// 响 应 单 击 "启动 动画 "按钮 
public void OnClickButton(View v) { 
myImageView. setVisibility(View.VISIBLE); 
// 逆 时 针 旋 转 30 度 , 即 飞 机 起 飞 
myImageView. setRotation( - 30); 
PointF myStartPoint = new PointF(0, 1400); 
PointF myMiddlePoint = new PointF(400, 300); 
PointF myEndPoint = new PointF(700, 300); 
TypeEvaluator < PointF > myPointEvaluator = new TypeEvaluator < PointF>() { 
(@ Override 
public PointF evaluate(float fraction, PointF startValue, PointF endValue) { 
float x = startValue.x + fraction x (endValue.x - startValue. x); 
float Y = startValue.y + fraction * (endValue.y - startValue.y); 
// 当 Y 值 达到 300 时 ,飞机 (图 像 ) 保 持 水 平 
if (Y == 300) 
myImageView. setRotation(0); 
PointF myEvaluatePoint = new PointF(x, y); 
return myEvaluatePoint; 
} }; 
// 运 动 (飞行 ) 路 径 为 myStartPoint，myMiddlePoint，myEndPoint 连 成 的 折线 
ValueAnimator myValueAnimator = ValueAnimator. ofObject(myPointEvaluator, 
myStartPoint, myMiddlePoint, myEndPoint); 
myValueAnimator. setDuration( 5000); 
myValueAnimator. setInterpolator(new DecelerateInterpolator( ) ); 
myValueAnimator. addUpdateListener(new ValueAnimator. AnimatorUpdateListener( ){ 
(Override 
public void onAnimationUpdate(ValueAnimator animation) { 
// 根 据 变化 的 路 径 值 设置 ImageView 的 坐标 
PointF myCurrentPoint = (PointF) animation. getRnimatedValue( ); 
myImageView. setX(myCurrentPoint. x); 
myImageView. setY(myCurrentPoint. y); 
} }); 
myValueAnimator. start( ); // 启 动 动 画 
} 


上 面 这 段 代 码 在 MyCode\ MySample853\app\src\ main\java\com\bin\luo\mysample\ 


MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample853 文件 夹 中 。 


133” 自 定义 TypeEvaluator 以 GIF 动画 显示 图 像 


此 实例 主要 通过 创建 自 定义 估 值 器 MyTypeEvaluator 配置 图 像 数组 索引 的 规则 ,并 使 用 该 自 定 
义 估 值 器 创建 ValueAnimator 动画 ,从 而 实现 以 GIF 动画 风格 显示 图 像 。 当 实例 运行 之 后 , 单 击 “ 启 
动 动画 ?按钮 , 则 将 以 GIF 动画 的 风格 显示 图 像 数 组 中 的 每 幅 图 像 ,效果 分 别 如 图 133. 1 的 左 图 和 碳 


图 所 示 。 
主要 代码 如 下 : 
public class MainActivity extends Activity { 


ImageView myImageView; 
int[ ] myImages = {R. mipmap. mygif layerl,R.mipmap. mygif layer?2, 
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R. mipmap. mygif layer3,R.mipmap.mygif layer4,R.mipmap.mygif layer5, 
R. mipmap. mygif layer6,R.mipmap.mygif layer7,R.mipmap.mygif layer8}; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity_main); 
myImageView = (ImageView)findViewById(R. id.myImageView); 
} 
public void onClickButton(View v) { // 响 应 单 击 " 启 动 动画 "按钮 
// 设 置 nyValueAnimator 动画 显示 的 图 像 数组 开始 索引 是 0, 结束 索引 是 7 
ValueAnimator myValueAnimator = 
ValueAnimator. ofObject (new MyTypeEvaluator( ), 0,7); 
myValueAnimator. addUpdateListener(new ValueAnimator. AnimatorUpdateListener(){ 
@ Override 
public void onAnimationUpdate(ValueAnimator animation) { 
int myIndex = (int)animation. getRhnimatedValue( ); 
// 根 据 时 间 线 确定 图 像 数组 索引 
myImageView. setImageResource(myImages[myIndex] ) ; 


只 


myValueAnimator. setDuration( 500); // 动 画 持续 时 间 是 500ms 
myValueAnimator. setRepeatCount( - 1); // 无 限 重复 
myValueAnimator. start( ); // 启 动 动画 


} 
// 自 定义 TypeEvaluator 制定 数组 索引 规则 
public class MyTypeEvaluator implements TypeEvaluator < Integer > { 
(Override 
public Integer evaluate(float fraction, Integer startValue, Integer endValue) { 
int myCurrentIndex = (int)(startValue + fraction * (endValue — startValue) ) ; 
return myCurrentIndex; 


| 


sl 让 多 
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图 133. 1 
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上 和 面 这 段 代 人 码 在 MyCode\ MySample860\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,myValueAnimator 二 ValueAnimator. ofObject (new 
MyTypeEvaluator(),0,7) 表 示 根 据 自 定义 类 MyTypeEvaluator 的 规则 按照 图 像 数 组 索引 顺序 在 
myIndex 一 (int)animation. getAnimatedValue() 中 获得 当前 显示 图 像 的 索引 ,然后 将 该 图 像 显示 出 
来 。 此 实例 的 完整 项 目 在 MyCode\MySample860 文件 夹 中 。 


134 使 用 Animation 实现 图 像 围 绕 自 身 中 心 诞 转 
此 实例 主要 通过 使 用 补 间 动 夯实 现 图 像 围绕 自 刁 中 心 旋 转 。 当 实例 运行 之 后 , 单 击 “开始 播放 动 


画 ” 按 钮 , 则 图 像 ( 车 轮 ) 将 围绕 自身 中 心 进行 旋转 ,如 图 134. 1 的 左 图 和 右 图 所 示 。 单 击 “ 停 止 播放 动 
画 ” 按 钮 , 则 停止 旋转 。 
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开始 播放 动画 停止 播放 动画 “|‖ 开始 播放 动画 。 停止 播放 动画 


图 134.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
Animation myAnimation; 
ImageView mylmageView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 


// 获 取 动 画 资源 
myAnimation = AnimationUtils. loadAnimation(this, R.anim.myanimation); 
myAnimation. setFillAfter(true); // 设 置 动 画 停 留 在 最 后 的 状态 


myImageView = (ImageView)findViewById(R. id.myImageView); 
Button myBtnPlay = (Button)findViewById(R. id. myBtnPlaV) ; 
myBtnPlay. setOnClickListener(new View. OnClickListener( ){ 
public void onClick(View arg0){ // 响 应 单 击 "开始 播放 动画 "按钮 
myImageView. startAnimation(myAnimation);  ” // 在 控件 上 实施 动画 
3 
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Button myBtnStop = (Button)findViewById(R. id. myBtnStop ) ; 
myBtnStop. setOnClickListener(new View. OnClickListener(){ 


public void onClick(View arg0){ // 响 应 单 击 " 停 止 播放 动画 "按钮 
myImageView. clearAnimation( ); // 在 控件 上 清除 动画 
3 
} } 


上 面 这 段 代 码 在 MyCode\ MySamplel50\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myAnimation 一 AnimationUtils. loadAnimation (this， 
R. anim. myanimation ) 用 于 加 载 在 res \anim 目录 中 的 动画 资源 myanimation。 mylmageView. 
startAnimation (myAnimation) 用 于 在 ImageView 控件 上 执行 myAnimation 动画 , mylmage. 
clearAnimation() 用 于 在 ImageView 控件 上 清除 动画 效果 。myanimation 动画 的 主要 内 容 如 下 : 


<?Xxml version= "1.0" encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android" 
android: interpolator = "android:anim/accelerate interpolator"> 
< rotate android: repeatMode = "restart" 
android: repeatCount = " infinite" 
android:fromDegrees = "0" 
android: toDegrees = "1800" 
android:pivotX="50%" 
android:pivotY= "50%" 
android: duration = "3000"/> 
</ set > 


上 面 这 段 代 码 在 MyCode\MySamplel50\app\src\main\res\anim\myanimation. xml 文件 中 。 在 
这 段 代 码 中 ,android:fromDegrees 王 "0" 表 示 旋 转动 画 的 开始 角度 。android: toDegrees 王 "1800" 表 
示 旋 转动 画 的 结束 角度 。android:pivotX 二 "50%" 和 android:pivotY 王 "50% "表示 旋转 中 心 的 水 平 
坐标 和 垂直 坐标 (以 图 像 中 心 为 旋转 中 心 ) 。android:duration 王 "3000" 表 示 动 画 的 持续 时 间 是 3 秒 。 
android:repeatMode 王 "restart" 表 示 动 男 的 重复 模式 是 重新 开始 。android:repeatCount 王 "infinite" 
表示 动画 的 重复 次 数 是 无 限 次 。 此 实例 的 完整 项 目 在 MyCode\MySamplel50 文件 夹 中 。 


135 ”上 自 定义 Animation 实现 许 转 切换 扑 死 牌 正 反 面 


此 实例 主要 通过 在 月 定义 类 Rotate3DAnimation 中 使 用 Camera 和 Matrix, 并 使 用 和 月 定义 类 
Rotate3DAnimationListener 动态 切换 指定 (扑克 牌 的 正 反 面 ) 图 像 ,实现 扑克 牌 正 反面 的 旋转 切换 。 
当 实 例 运 行 之 后 , 单 击 “ 显 示 正 面 ” 按 钮 , 则 扑克 牌 反 面 图像 将 围绕 y 轴 旋 转 180 ,切换 到 扑克 有 牌 正面 ， 
如 图 135. 1 的 左 图 所 示 。 单 击 “ 显 示 反 面 ” 按 钮 , 则 扑克 有 牌 正面 图 像 将 围绕 y 轴 旋 转 180 ,切换 到 扑克 
牌 反面 ,如 图 135. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickBtnl (View v) { // 响 应 单 击 " 显 示 正 面 "按钮 
myRotate3DAnimation = new Rotate3DAnimation(180, 90); // 开 始 前 半 段 旋转 动画 
myRotate3DAnimation. setAnimationListener(new Rotate3DRnimationListener() { 
(Override 
public void onAnimationEnd(Animation animation) { 
// 切 换 显示 (扑克 牌 正面 ) 图 像 


myImageView. setImageDrawable(getDrawable(R.mipmap.mybmpfront) ); 


// 重 新 初始 化 对 象 , 以 重 置 旋转 参数 
myRotate3DAnimation = new Rotate3DAnimation(90, 0); 
myImageView. startAnimation(myRotate3DAnimation); // 执 行 后 半 段 动画 
} }); 
myImageView. startAnimation(myRotate3DAnimation); // 启 动 动画 
} 
public void onClickBtn2(View v) { // 响 应 单 击 "显示 反面 "按钮 


myRotate3DAnimation = new Rotate3DAnimation(0, 90); 
myRotate3DAnimation. setAnimationListener(new Rotate3DAnimationListener() { 
(Override 
public void onAnimationEnd(Animation animation) { 
// 切 换 显 示 ( 扑 克 牌 反面 ) 图 像 
myImageView. setImageDrawable(getDrawable(R. mipmap. mybmpback) ); 
// 重 新 初始 化 对 象 , 以 重 置 旋转 参数 
myRotate3DAnimation = new Rotate3DAnimation(90, 180); 
myImageView. startAnimation(myRotate3DAnimation); // 启 动 后 半 段 动画 
} }); 
myImageView. startAnimation(myRotate3DAnimation); // 启 动 动画 
} 
public class Rotate3DAnimation extends Animation { 


int myCenterX，myCenterY; // 旋 转 中 心 位 置 坐标 
Camera myCamera = new Cameral( ); 
int myEndDegrees, myStartDegrees; // 旋 转 前 后 的 角度 


public Rotate3DAnimation( int startDegrees, int endDegrees) { 
myStartDegrees = startDegrees; 
myEndDegrees = endDegrees; 

} 

(Override 

public void initialize(int width, int height, int parentWidth, int ParentHeight){ 
super. initialize(width, height, parentWidth, parentHeight); 


myCenterX = width / 2; // 设 置 旋转 中 心 位 置 坐 标 值 
myCenterY = height / 2; 
setFillAfter(true); // 设 置 在 动画 完成 后 是 否 保留 状态 
setDuration(2000); // 设 置 动 画 时 长 2000ms 
// 设 置 先 加 速 后 减速 的 时 间 插 值 器 
setInterpolator(new AccelerateDecelerateInterpolator( ) ) ; 
} 
(Override 
protected void applyTransformation(float interpolatedTime, Transformation 七 ) { 
Matrix myMatrix = 七 .getMatrix() ; // 获 取 和 矩阵 对 象 
myCamera. Savel( ) ; 
myCamera. rotateY(myStartDegrees ) ; // 设 置 旋转 前 角度 
myCamera. rotateY( (myEndDegrees — myStartDegrees) * interpolatedTime); 
myCamera. getMatrix(myMatrix); // 获 取 变 换 和 矩阵 


myMatrix. preTranslate( - myCenterX, — myCenterY); 
myMatrix. postTranslate(myCenterX, myCenterY); 
myCamera. restore( ) ; 
是 
public class Rotate3DAnimationListener implements Animation. AnimationListener { 
(Override 
public void onAnimationStart(Animation animation){ } 
(Override 
public void onanimationEnd(RAnimation animation){ } 
(Override 
public void onAnimationRepeat(Animation animation){ } 
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上 F 面 这 段 代 码 在 MyCode\ MySample871 \app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample871 文件 夹 中 ， 
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图 135.1 


136 ”使 用 AnimationSet 实现 组 合 多 个 不 同 的 动画 


此 实例 主要 通过 使 用 AnimationSet 的 playTogether() 方 法 登 加 多 个 动画 ,实现 多 个 动画 同时 执 
行 的 效果 。 当 实例 运行 之 后 , 单 击 “ 开 始 播放 动画 ?按钮 , 则 图 像 将 在 3 秒 内 同时 执行 缩放 和 旋转 的 动 
作 ,效果 分 别 如 图 136. 1 的 左 图 和 右 图 所 示 。 


Ar 
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图 136. 1 


public class MainActivity extends Activity { 
ImageView mylImageView; 
AnimatorSet myAnimatorSet; 
(QOverride 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
ObjectAnimator myObjectAnimatorl1 = 
ObjectAnimator. ofFloat(myImageView, "scaleXx", 1.0f, 3.0f, 1.0f); 
ObjectAnimator myObjectAnimator2 = 
ObjectAnimator. ofFloat(myImageView, "scaleY", 1.0f, 3.0f, 1.0f); 
ObjectAnimator myObjectAnimator3 = 
ObjectAnimator. ofFloat(myImageView, "rotationY", 0.0f, 720.0f); 
myAnimatorSet = new AnimatorSet( ); 
myAnimatorSet. playTogether(myObjectAnimator1l, myObjectAnimator2, 


myObjectAnimator3); 
myAnimatorSet. setDuration( 3000); 
myAnimatorSet. start( ); 
} 
// 响 应 单 击 " 开 始 播放 动画 "按钮 
public void onClickButton]l (View v){ myAnimatorSet. start(); } 
// 响 应 单 击 "停止 播放 动画 "按钮 
public void onClickButton2(View v){ myAnimatorSet. cancel(); } 


上 和 面 这 段 代 人 码 在 MyCode\ MySamplel71 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myObjectAnimatorl 三 ObjectAnimator. ofFloat 
(mylImageView, "scaleX", 1.0f,3. 0f, 1. 0f) 表示 在 指定 的 时 间 内 ,myImageView 控件 在 水 平方 回 
放大 3 倍 , 然 后 还 厚 。myObjectAnimator2 三 ObjectAnimator. ofFloat (myImageView，"scaleY "， 
1. 0f，3. 0f，1. 0f) 表示 在 指定 的 时 间 内 ,myImageView 控件 在 垂直 方向 放大 3 倍 , 然 后 还 厚 。 
myObjectAnimator3 = ObjectAnimator. ofFloat (myImageView,， "rotationY", 0. 0f, 720. 0f) 表 示 在 
指定 的 时 间 内 ,myImageView 控件 围绕 yy 轴 旋 转 720 度 。myAnimatorSet. playTogether 
(myObjectAnimatorl ， myObjectAnimator2， myObjectAnimator3 ) 表示 将 myAnimatorl， 
myAnimator2 ,myAnimator3 三 个 动画 加 载 到 myAnimatorSet 动画 集合 中 ,每 个 动画 的 所 有 动作 ( 旋 
转 和 缩放 ) 在 指定 的 时 间 内 同时 执行 。 此 实例 的 完整 项 目 在 MyCode\MySamplel71 文件 夹 中 。 


137 使 用 Animation 实现 按照 顺序 显示 网 格 Item 


此 实例 主要 通过 使 用 GridLayoutAnimationController 加 载 XML 文件 中 的 透明 度 改变 动画 , 实 
现 按照 指定 的 顺序 和 方向 在 GridView 的 每 个 单元 格 (Item) 中 实现 透明 度 从 无 到 有 显示 动画 。 当 实 
例 运行 之 后 ,将 按照 从 右 到 左 、 从 下 向 上 的 顺序 逐个 显示 在 GridView 的 每 个 单元 格 中 的 图 像 ,效果 分 
别 如 图 137. 1 的 左 图 和 右 图 所 示 。 


CS> Androiaig 本 应 有 200 。 实战 篇 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

(Override 

public void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity_main); 
GridView myGridView = (GridView) findViewById(R. id.myGridView); 
ArrayList < HashMap < String, Object >> myArray = new ArrayList <>(); 
HashMap < String, Object > myItem = new HashMap <>(); 
// 为 节省 篇 幅 , 此 处 省 略 了 加 载 Item 代码 ,详细 内 容 请 看 源 文件 


SimpleAdapter myAdapter = new SimpleAdapter(this, myArray, 
R. layout. myitem, new String[ ]{"myImage", "myName" } ， 
new int[ ]{R. id. myImage,R. id. myName} ) ; 

myGridView. setAdapter(myAdapter); 


Animation myAnimation = AnimationUtils. loadAnimation(this, R.anim.myanim); 


GridLayoutAnimationController myGridLayoutAnimationController = 


new GridLayoutAnimationController(myAnimation); 


// 设 置 动 画 方向 为 从 右 向 左 、 从 下 向 上 
myGridLayoutAnimationController. setDirection( 


GridLayoutAnimationController.DIRECTION BOTTOM TO TOP| 
GridLayoutAnimationController.DIRECTION RIGHT TO LEFT); 


myGridLayoutAnimationController. setDirectionPriority( 
GridLayoutAnimationController. PRIORITY ROW); 
// 在 myGridView 控件 上 设置 动画 
myGridView. setLayoutAnimation(myGridLayoutAnimationController); 
} } 
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图 137.1 


// 设 置 行 优先 级 


上 和 面 这 段 代码 在 MyCode\MySample862\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代码 中 , Animation myAnimation 一 AnimationUtils. loadAnimation(this，R. anim. 


myanim) 用 于 根据 myanim. xml 文件 创建 透明 度 动画 ,GridVievw 控件 的 (单元 格 ) 每 个 Item 的 透明 度 改变 
动画 则 由 myAnimation 实现 ,myanim. xml 文件 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 
< Set xmlns:android = "http://schemas. android. com/apk/res/android"> 
< alpha android: duration = "1500" 
android:fromAlpha = "0.0" 
android: interpolator = " (Wandroid: anim/accelerate interpolator" 
android: toAlpha = "1.0" /> 
</set > 


上 面 这 段 代 码 在 MyCode\MySample862\app\src\main\res\anim\myanim. xml 文件 中 。 在 这 上段 
代码 中 ,android:fromAlpha 二 "0.0" 表 示 在 动画 开始 时 完全 透明 ,android:toAlpha 二 "1.0" 表 示 在 动 
画 结 束 时 完全 不 透明 ,androld:duration 王 "1500" 表 示 动 画 持 续 时 间 是 1500ms。 此 实例 的 完整 项 目 在 
MyCode\MySample862 文件 夹 中 。 


138 ”使 用 windowAnimations 实现 缩放 对 话 框 窗口 


此 实例 主要 通过 在 以 DialogFragment 类 为 基 类 的 自 定义 类 MyDialogFragment 中 重 写 onStart() 
方法 ,实现 在 弹出 或 关闭 提示 框 时 显示 缩放 窗口 动画 。 当 实例 运行 之 后 ,直接 按 下 后 退 键 , 则 将 由 小 
到 大 弹出 一 个 提示 框 , 单 击 该 提示 框 中 的 “返回 ”按钮 , 则 提示 框 将 由 大 到 小 ,直到 消失 ,效果 分 别 如 
图 138. 1 的 左 图 和 右 图 所 示 。 


来 自 应 用 的 提示 信息 : 


确定 要 退出 么 ? 


138. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 


> 
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super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
} 


(@ Override 
public boolean onKeyDown( int keyCode, KeyEvent event){ 
if(event. getKeyCode() == KeyEvent. KEYCODE_ BACK) { // 按 下 后 退 键 显 示 对 话 框 


MyDialogFragment myDialogFragment = new MyDialogFragment( ); 
myDialogFragment. show(getFragmentManager( )," MyDialog" ); 
return true; 
} 
return super. onKeyDown(keyCode, event ); 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample743\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myDialogFragment 一 new MyDialogFragment( ) 中 的 
MyDialogFragment 是 自 定义 类 ,用 于 实现 缩放 窗口 ,该 类 的 主要 代码 如 下 : 


public class MyDialogFragment extends DialogFragment { 
(Override 
public void onStart() { 
super. onStart(); 
Window myWindow = getDialog( ). getWindow( ); 
WindowManager. LayoutParams myAttrs = myWindow. getRttributes(); 
myAttrs. windowRnimations = R.style.MyDialogAnim; 
myWindow. setAttributes(myAttrs); 
} 
(Override 
public Dialog onCreateDialog(Bundle savedInstanceState) { 
AlertDialog. Builder myBuilder = new AlertDialog. Builder(getActivity()); 
myBuilder. setTitle( "来 自 应 用 的 提示 信息 :"); 
myBuilder. setMessage( "确定 要 退出 么 ?"); 
myBuilder. setPositiveButton( "确定 ",new DialogInterface. OnClickListener( ){ 
(Override 
public void onClick(DialogInterface dialog, int which) { 
getActivity().finish( ); 
dialog. dismiss( ); 
} 1); 
myBuilder. setNegativeButton(" 返 回 "，new DialogInterface. OnClickListener() { 
(Override 
public void onClick(DialogInterface dialog, int which){dialog. dismiss(); } 
}); 
return myBuilder. create( ) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySample743\app\src\ main\java\com\bin\luo\mysample\ 
MyDialogFragment. java 文件 中 。 在 这 段 代 码 中 ，myAttrs. windowAnimations 三 R. style. 
MyDialogAnim 表示 提示 框 在 显示 或 关闭 时 使 用 R. style. MyDialogAnim 指定 的 动画 。 
MyDialogAnim 在 Styles 中 指定 ,主要 代码 如 下 : 


< reSources > 
< style name = "AppTheme" parent = "android:Theme. Material. Light. DarkRctionBar" /> 
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< style name = "MyDialogAnim"> 
< item name = " (android:windowEnterRnimation">(Oanim/myenter </ item > 
< item name = " (android:windowExitRnimation">(Oanim/myexit </item > 
</ style > 
</resources > 


上 面 这 段 代 码 在 MyCode\MySample743\app\src\main\res\values\styles. xml 文件 中 。 在 这 段 
代码 中 ,< item name 王 "android:windowEnterAnimation">@animy/myenter </item > 表示 进 场 动画 
使 用 myenter 动画 。< item name 一 "android: windowExitAnimation"> (@anim/myexit </item > 表 


示 退 场 动画 使 用 myexit 动画 。myenter 动画 文件 的 主要 内 容 如 下 : 


<?xml version = "1.0”encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android"> 
< Scale android: duration = "5000" 
android:fromXScale = "0.0" 
android: fromYScale = "0.0" 
android:pivotX="50%" 
android:pivotY= "50%" 
android:toXScale = "1.0" 
android:toYScale = "1.0" /> 
</set > 


上 面 这 段 代 码 在 MyCode\MySample743\app\src\main\res\anim\myenter. xml 文件 中 。myexit 
动画 文件 的 主要 内 容 如 下 : 


<?xml] version = "1.0" encoding = "utf 一 8"?> 
< Set xmlns:android = "http://schemas.android. com/apk/res/android"> 
< Scale android: duration = "5000" 
android: fromXScale = "1.0" 
android:fromYScale = "1.0" 
android:pivotX="50%" 
android:pivotY= "50%" 
android:toXScale = "0.0" 
android: toYScale = "0.0"/> 
</set > 


上 面 这 段 代 码 在 MyCode\MySample743\app\src\main\res\anim\myexit. xml 文件 中 。 此 实例 
的 完整 项 目 在 MyCode\MySample743 文件 夹 中 。 


139 ”使 用 AnimationDrawable 播放 多 幅 图 像 


此 实例 主要 通过 使 用 AnimationDrawable 存储 多 幅 图 像 ,并 设置 每 幅 图 像 之 间 的 间隔 时 长 ,实现 
类 似 于 GIF 格式 的 动 图 效果 。 当 实例 运行 之 后 , 单 击 “开始 播放 ?按钮 , 则 开始 逐 帧 播放 多 幅 图 像 ; 单 
击 “ 停 止 播放 ”按钮 , 则 停止 播放 图 像 , 效 果 分 别 如 图 139. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 

public class MainActivity extends Activity { 


AnimationDrawable myMAnimationDrawable; 
Override 
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protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 


setContentView(R. layout.activity main); 


ImageView myImageView = 


(ImageView) findViewById(R. id. myImageView) ; 


myAnimationDrawable = new RnimationDrawable( ); 


Drawable myDrawablel 
Drawable myDrawable2 = 
Drawable myDrawable3 = 
Drawable myDrawable4 = 
Drawable myDrawable5 = 
Drawable myDrawable6 = 
Drawable myDrawable7 = 
Drawable myDrawable8 = 


= getResources().getDrawable(R.mipmap. mybmp001 ) ; 


getResources( ) . getDrawable(R. mipmap. mybmp002); 
getResources().getDrawable(R. mipmap. mybmp003 ) ; 
getResources( ) . getDrawable(R.mipmap.mybmp004); 
getResources( ) . getDrawable(R.mipmap.mybmp005 ); 
getResources().getDrawable(R.mipmap. mybmp006); 
getResources().getDrawable(R. mipmap. mybmp007 ) ; 
getResources( ).getDrawable(R. mipmap. mybmp008); 


// 将 图 像 添加 至 AnimationDrawable 对 象 中 ,并 设 定 显 示 时 长 
myAnimationDrawable. addFrame( myDrawablel, 100); 
myAnimationDrawable. addFrame( myDrawable2, 100); 
myAnimationDrawable. addFrame( myDrawable3, 1000); 
myAnimationDrawable. addFrame(myDrawable4, 100); 
myAnimationDrawable. addFrame( myDrawable5, 100); 
myAnimationDrawable. addFrame( myDrawable6, 100); 
myAnimationDrawable. addFrame( myDrawable7, 100); 
myAnimationDrawable. addFrame( myDrawable8, 100); 
myAnimationDrawable. setOneShot (false); // 一 直播 放 该 动画 
// 在 ImageView 控件 上 应 用 该 AnimationDrawable 对 象 
myImageView. setImageDrawable( myAnimationDrawable); 


} 


// 啊 应 单 击 "开始 播放 "按钮 
public void onClickButtonl (View v) {myAnimationDrawable. start( ); } 


// 响 应 单 击 "停止 播放 "按钮 
public void onClickButton2(View v) {myanimationDrawable. stop( );} 


MySample MySample 
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上 面 这 段 代 码 在 MyCode\ MySample771 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,myAnimationDrawable. addFrame(myDrawablel ，100) 表 示 将 
myDrawablel( 一 幅 图 像 ) 添 加 到 myAnimationDrawable 中 ,并 且 停 留 100ms。myAnimationDrawable. 
setOneShot(false) 表 示 一 直播 放 该 动画 ,如 果 参 数值 为 true, 则 在 播放 一 遍 后 停止 。 此 实例 的 完整 项 
日 在 MyCode\MySample771 文件 夹 中 。 


140 使 用 AnimationDrawable 创建 爆炸 动 团 


此 实例 主要 通过 使 用 AnimationDrawable 管理 动画 ,并 在 FrameLayout 的 OnTouch 事件 啊 应 方 
法 中 控制 动画 的 显示 位 置 ,实现 单 击 屏 幕 任意 位 置 即 可 实现 播放 爆炸 动画 。 当 实例 运行 之 后 , 单 击 屏 
幕 任意 位 置 即 可 播放 爆炸 动画 ,效果 分 别 如 图 140. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

private MyView myView; 

private AnimationDrawable myAnimationDrawable; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
FrameLayout myFrameLayout = new FrameLayout(this); 
myFrameLayout. setBackgroundColor( Color. WHITE ) ; 
setContentView(myFrameLayout); 
myView = new MyView(this); 


myView. setBackgroundResource(R. drawable. mybmps ); // 设 置 myView 显示 爆炸 动画 
myView. setVisibility(View. INVISIBLE); // 设 置 myView 默认 隐藏 


myAnimationDrawable = (RnimationDrawable)myView. getBackground( ) ; 
myFrameLayout. addView(myView): 
myFrameLayout. setOnTouchListener(new View. OnTouchListener( ){ 
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(@ Override 
public boolean onTouch(View source, MotionEvent event){ 
if(event.getAction() == MotionEvent.ACTION DOWN){ 
myAnimationDrawable. stop( ); 
float x = event.getX(); 
float y = event. getY( ); 
myView. setLocation((int)y ~ 90 , (int)x- 90); 
myView. setVisibility(View. VISIBLE); 
myAnimationDrawable. start( ); 
} 
return false; 
Ld 
class MyView extends ImageView{ 
public MyView(Context context) { super(context); } 
public void setLocation( int top, int left){ 
this. setFrame( left, top, left + 180, top + 180); 
} 
(Override 
protected void onDraw(Canvas canvas){ super. onDraw(canvas); } 


} } 


// 停 止 动画 播放 


// 控 制 动 画 的 显示 位 置 


// 开 始 动画 播放 


// 自 定义 View, 用 于 承载 爆炸 动画 


// 控 制 MyView 的 显示 位 置 


上 面 这 段 代 码 在 MyCode\ MySamplel47\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myView. setBackgroundResource(R. drawable. mybmps) 
用 于 将 动画 资源 mybmps 设置 为 myView 的 背景 资源 。mybmps 是 负责 配置 动画 的 XML 文件 ,其 主 
要 代码 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<animation— list xmlns:android = "http://schemas. android. com/apk/res/android" 
android: oneshot = "true"> 
< item android:drawable = "(@drawable/mybmp001" android:duration = "60" /> 


< item android:drawable = "drawable/mybmp002" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp003" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp004" android: duration 1 
< item android:drawable = "(@drawable/mybmp005" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp006" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp007" android:duration = "60" /> 
< item android:drawable = "drawable/mybmp008" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp009" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp010”android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp011" android:duration "0" 
< item android:drawable = "(@drawable/mybmp012" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp013" android:duration = "60" /> 
< item android:drawable = "(@drawable/mybmp014" android:duration = "60" /> 


</animation— list> 


上 面 这 段 代 码 在 MyCode\MySamplel47\app\src\main\res\drawable\mybmps. xml 文件 中 。 在 
这 段 代码 中 ,< animation-list > 元 素 为 根 节 点 ,< item > 节点 定义 了 每 一 帧 的 参数 ,表示 一 个 drawable 
资源 的 帧 (动画 图 像 ) 和 帧 间隔 。 设 置 android:oneshot 属性 为 true, 表 示 此 动画 只 执行 一 次 ,并 停留 
在 最 后 一 帧 。 设 置 android:oneshot 为 false, 则 表示 动画 循环 播放 。 此 实例 的 完整 项 目 在 MyCode\ 
MySamplel47 文件 夹 中 ， 
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141 使 用 RotateAnimation 实现 围绕 自身 中 心 诞 转 
此 实例 主要 通过 使 用 RotateAnimation 创建 旋转 动画 ,实现 图 像 以 自 号 为 中 心 进行 旋 转 。 当 实例 


运行 之 后 , 单 击 “开始 播放 动画 ”按钮 , 则 图 像 将 以 顺 时 针 方 钻 进 行 旋 转 , 且 永 不 停 软 ,效果 分 别 如 
图 141. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


开始 播放 动画 停止 播放 动画 开始 播放 动画 停止 播放 动画 


图 141. 1 


public class MainActivity extends Activity { 

ImageView myImageView; 

RotateAnimation myRotateAnimation; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myImageView = (ImageView) findViewById(R. id. myImageView) ; 
myImageView. setVisibility(View. GONE); 


myRotateAnimation = new RotateAnimation(0, 360, 540, 960); // 创 建 旋转 动画 

myRotateAnimation. setDuration( 3000); // 设 置 动画 持续 时 间 3 秒 

myRotateAnimation. setRepeatCount(Rnimation. INFINITE); // 设 置 动 画 无 限 次 重复 

myImageView. setAnimation(myRotateAnimation); // 在 ImageView 图 像 控 件 上 添加 动画 
} 

// 响 应 单 击 " 开 始 播放 动画 "按钮 


public void onClickButtonl (View v) { 
myImageView. setVisibility(View.VISIBLE); 
myRotateAnimation. startNow( ); 
} 
// 响 应 单 击 "停止 播放 动画 "按钮 
public void onClickButton2(View v) {myRotateAnimation. cancel(); } 
} 
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上 面 这 上段 代码 在 MyCode\ MySamplel58\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myRotateAnimation 三 new RotateAnimation (0，360， 
540，960) 用 于 创建 一 个 旋转 动画 ,RotateAnimation() 方 法 的 语法 声明 如 下 : 


RotateAnimation(float fromDegrees, 
float toDegrees, float pivotX, float pivotY) 


其 中 ,float fromDegrees 表示 开始 角度 。float toDegrees 表示 结束 角度 。float pivotX 和 float 
pivotY 表示 旋转 中 心 点 的 坐标 ,pivotX 为 距离 左 侧 的 偏 移 量 ,pivotY 为 距离 项 部 的 偏 移 量 , 即 为 相对 
于 View 左上 角 (0,0) 的 坐标 。 如 果 View 的 width 是 100px, height 是 100px, 则 RotateAnimation 
(0,10,100,100) 将 以 右 下 角 顶 点 为 旋转 中 心 点 ,从 原始 位 置 顺 时 针 旋 转 10 度 ; RotateAnimation(0,10,0， 
0) 将 以 View 的 左上 角 为 旋转 中 心 点 ,旋转 10”"。 此 实例 的 完整 项 目 在 MyCode\MySamplel58 文件 夹 中 。 


142 使 用 AlphaAnimation 创建 深入 淡出 动画 


此 实例 主要 通过 使 用 AlphaAnimation 控制 图 像 的 透明 度 ,实现 以 动画 的 形式 淡 和 淡出 地 显示 或 
隐藏 图 像 。 当 实例 运行 之 后 , 单 击 “ 播 放 淡 入 动画 ”按钮 , 则 图 像 ( 妖 怪 ) 将 以 淡 入 的 动画 风格 在 5 秒 内 
显示 出 来 ,如 图 142. 1 的 左 图 所 示 。 单 击 “ 播 放 淡 出 动画 ”按钮 , 则 图 像 ( 妖 怪 ) 将 以 淡出 的 动画 风格 在 
5 秒 内 消失 ,如 图 142. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


外 ~ 


MySample MySample 


播放 淡 入 动画 播放 淡出 动画 播放 淡 入 动画 播放 淡出 动画 


public class MainActivity extends Activity { 
AlphaAnimation myAnimationIn,myAnimationOut; 
ImageView myImageView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
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setContentView(R. layout.activity main); 

// 配 置 淡 入 动画 

myAnimationIn = new AlphaAnimation(0.0f, 1.0f); 
myAnimationIn. setDuration( 5000); 

myAnimationIn. setFillAfter(true); 

// 配 置 淡出 动画 

myAnimationOut = new AlphaAnimation(1.0f, 0.0f); 
myAnimationOut. setDuration(5000); 

myAnimationOut. setFillAfter(true); 

myImageView = (ImageView)findViewById(R. id.myImageView); 


} 

public void onClickButtonl (View v) { // 响 应 单 击 " 播 放 淡 入 动画 "按钮 
myImageView. clearAnimation( ); // 在 myImageView 上 清除 动画 
// 在 myImageView 上 实施 淡 入 动画 
myImageView. startAnimation(MainActivity. this. myAnimationIn); 

} 

public void onClickButton2(View v) { // 响 应 单 击 "播放 淡出 动画 "按钮 
myImageView. clearAnimation( ); // 在 myImageView 上 清除 动画 
// 在 myImageView 上 实施 淡出 动画 
myImageView. startAnimation(MainActivity. this. myAnimationOut ); 

国 ， 


上 F 面 这 段 代 码 在 MyCode\ MySamplel51 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myAnimationIn 二 new AlphaAnimation(0. 0f，1. 0f) 用 
于 构建 一 个 透明 度 动画 ,参数 0. 0f 为 动画 开始 参数 ,表示 完全 透明 ; 参数 1. 0f 为 动画 结束 参数 ,表示 
完全 不 透明 。myAnimationIn. setDuration (5000) 表示 动画 持续 时 间 为 5 秒 。myAnimationIn. 
setFillAfter(true) 表 示 在 动画 结束 时 保持 最 后 的 状态 。 此 实例 的 完整 项 目 在 MyCode\MySamplel51 
文件 夹 中 。 


143 ”使 用 ScaleAnimation 创建 缩放 图 像 动 画 


此 实例 主要 通过 使 用 ScaleAnimation 创建 缩放 动画 ,从 而 使 图 像 以 自 号 为 中 心 在 水 平方 喇 和 垂 
直方 向 上 进行 缩放 。 当 实例 运行 之 后 , 单 击 “开始 播放 动画 ?按钮 , 则 图 像 将 以 自身 为 中 心 进行 放大 ， 
效果 分 别 如 图 143. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


// 啊 应 单 击 "开始 播放 动画 "按钮 
public void onClickButtonl (View v) { 
// 清 除 myImageView 控件 的 动画 
myImageView. clearAnimation( ); 
myScaleAnimation = new ScaleAnimation(0.0f, 1.4f, 0.0f, 1.4f, 
Rnimation. RELATIVE TO SELF, 0.5f, Animation.RELATIVE TO SELF, 0.5f); 
// 设 置 动画 持续 时 间 3 秒 
myScaleAnimation. setDuration( 3000); 
// 在 myImageView 控件 上 添加 动画 
myImageView. setAnimation(myScaleAnimation); 
myScaleAnimation. start( ); 
} 
// 响 应 单 击 "停止 播放 动画 "按钮 
public void onClickButton2(View v) { myScaleAnimation. cancel(); } 
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MySample 


开始 播放 动画 停止 播放 动画 开始 播放 动画 停止 播放 动画 


图 143. 1 


上 F 面 这 段 代 码 在 MyCode\ MySamplel59\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myScaleAnimation 一 new ScaleAnimation(0. 0{f, 1. 4f，, 
0. 0f, 1.4{f, Animation. RELATIVE TO SELF, 0.5f{f, Animation. RELATIVE TO SELF, 0. 5f) 用 
于 创建 一 个 缩放 动画 ,ScaleAnimation() 构 造 函 数 的 语法 声明 如 下 : 


public ScaleAnimation(float fromxX, float toxX, float fromY, float toY, 
int pivotXType, float pivotXValue, int pivotYType, float pivotYValue) 


其 中 ,参数 float fromX 表示 xz 坐标 上 的 起 始 尺 寸 。 参 数 float toX 表示 x 坐标 上 的 结束 尺寸 。 参 
数 float fromY 表示 y 坐标 上 的 起 始 尺 寸 。 参 数 float toY 表示 y 坐标 上 的 结束 尺寸 。 参 数 int 
pivotXType 表示 xz 轴 的 伸缩 模式 ,可 以 取 值 为 ABSOLUTE、RELATIVE_TO_SELF、RELATIVE_ 
TO_PARENT。 参 数 float pivotXValue 表示 工 坐标 的 伸缩 值 。 参 数 int pivotYType 表示 > 轴 的 伸缩 
模式 ,可 以 取 值 为 ABSOLUTE RELATIVE TO_SELF RELATIVE TO_PARENT。 参 数 float 
pivotYValue 表示 y 坐标 的 伸缩 值 。 此 实例 的 完整 项 目 在 MyCode\MySample159 文件 夹 中 ， 


144 在 ViewPager 中 实现 上 下 请 动 的 转 场 动画 


此 实例 主要 通过 使 用 ViewPager. PageTransformer 接口 创建 自 定 义 转 场 动画 类 
VerticalPageTransformer, 从 而 在 ViewPager 中 实现 以 上 滑 下 滑 的 转 场 动画 切换 页 面 。 当 实例 运行 
之 后 ,如 果 手 指 在 屏幕 上 从 左 回 右 滑动 , 则 上 一 张 图 像 将 从 上 回 下 滑 入 ; 如 果 手 指 在 屏幕 上 从 右 回 左 
滑动 , 则 下 一 张 图 像 将 从 下 向 上 滑 入 ,效果 分 别 如 图 144. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


private class VerticalPageTransformer implements ViewPager. PageTransformer { 
(OOverride 
public void transformPage(View view, float position) { 
if (Position <= 1) { 
view. setTranslationX(view.getWidth() * - position); 
float myPosition = position *x view.getHeight(); 
view. setTranslationY(myPosition); // 在 垂直 方向 上 平移 图 像 
pj 
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上 F 面 这 段 代 码 在 MyCode\ MySample521 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,在 Android Studio 中 ,使 用 此 实例 的 相关 控件 需要 在 
gradle 中 引入 compile 'com. android. support:design:23. 3.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode 
\MySample521 文件 夹 中 。 


145 ”通过 下 拉手 指 实现 两 个 Activity 的 相互 切换 


此 实例 主要 通过 在 onTouchEvent() 事 件 啊 应 方法 中 计算 手指 在 按 下 和 抬 起 之 间 移 动 的 垂直 距 
离 ,判断 手指 是 否 执行 了 下 拉动 作 ,并 据 此 从 一 个 Activity 跳 转 到 另 一 个 Activity。 当 实例 运行 之 后 ， 
MainActivity 如 图 145. 1 的 左 图 所 示 ; 如 果 手 指 在 MainActivity 回 下 滑动 , 则 将 跳 转 到 NextActivity, 如 
图 145. 1 的 右 图 所 示 。 


SS 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

// 手 指 按 下 的 点 为 (x1，y1), 手 指 离开 的 点 为 (x2,y2) 

float xzl = 0,x2 = 0,yl = 0,y2 = 0; 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 

} 


(@ Override 
public boolean onTouchEvent (MotionEvent event) { 
if (event.getAction() == MotionEvent.ACTION DOWN) { // 当 手指 按 下 的 时 候 


xl = event.getX(); 

yl = event.getY(); 

} 

if (event.getAction() == MotionEvent.ACTION UP) { // 当 手指 离开 的 时 候 

x2 = event.getX(); 

Y2 = event.getY(); 

if (yl - 只 >50) { 
//Toast. makeText(MainActivity.this, "向 上 滑 ", Toast. LENGTH_ SHORT). show( ); 
} else if (y2 - yl > 50)f{ 
//Toast. makeText (MainActivity.this, "向 下 滑 ", Toast.LENGTH SHORT). show( ); 
// 在 下 滑 手 指 时 实现 由 MainActivity 切换 到 NextActivity 
Intent myIntent = new Intent(MainActivity.this,NextActivity.class); 
startActivity(myIntent ); 

} else if (x1 - x2 > 50) 1{ 
Toast. makeText (MainActivity.this, "向 左 滑 ", Toast. LENGTH SHORT). show( ); 

} else if (x2 - x1> 50) { 
Toast. makeText (MainActivity.this, "向 右 滑 ",， Toast. LENGTH SHORT) . show( ); 

} } 

return true; 

Fy 


上 面 这 段 代 码 在 MyCode\ MySample425 \app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,onTouchEvent() 方 法 用 于 啊 应 手指 在 屏幕 上 的 滑动 ， 
y2 一 yl 二 50 表 示 手 指 在 垂直 方 回 滑动 的 距离 大 于 50 像素 。myIntent 二 new Intent 
(MainActivity. this,NextActivity. class) 表 示 在 执行 startActivity(intent) 方 法 后 ,将 从 MainActivity 
跳 转 到 NextActivity。 此 外 需要 注意 的 是 , 当 在 应 用 中 新 增 NextActivity 之 后 ,需要 在 
AndroidManifest. xml 文件 中 添加 < activity android:name 三 ". NextActivity" /> 代码 进行 注册 。 此 实 
例 的 完整 项 目 在 MyCode\MySample425 文件 夹 中 。 


146 ”在 应 用 启动 时 使 用 进 场 动画 启动 Activity 
此 实例 主要 通过 在 onCreate() 方 法 中 调用 overridePendingTransition() 方 法 ,实现 在 启动 应 用 时 


显示 进 场 动 画 。 当 实例 运行 之 后 , 单 击 桌面 上 的 当前 应 用 图 标 , 则 应 用 将 从 小 到 大 膨胀 ,如 图 146. 1 
的 左 图 所 示 ; 直到 铺 满 整个 屏幕 ,如 图 146. 1 的 右 图 所 示 。 


MySample 


MySample 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
OverridePendingTransition(R. anim. myenter,R. anim. myexit); 
setContentView(R. layout. activity main); 


} 


上 面 这 上段 代码 在 MyCode\ MySample509\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,overridePendingTransition (R. anim. myenter, R. anim. 
myexit) 表 示 在 启动 应 用 时 执行 转 场 动画 myenter 和 myexit,myenter 动画 ( 进 场 动画 ) 的 作用 是 使 当 
前 应 用 窗口 由 小 变 大 ,myexit 动画 (退场 动画 ) 的 作用 是 使 手机 桌面 窗口 由 大 变 小 。myenter 动画 文 
件 的 主要 内 容 如 下 : 


<?xml version = "1.0" encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android"> 
< Scale android: duration = "5000" 
android: fromXScale = "0.0" 
android: fromYScale = "0.0" 
android:pivotX="50%" 
android:pivotY="50%" 
android:toXScale = "1.0" 
android:toYScale = "1.0"/> 
</set > 


上 面 这 段 代 码 在 MyCode\MySample509\app\src\main\res\anim\myenter. xml 文件 中 。myexit 
动画 文件 的 主要 内 容 如 下 : 
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<?Xxml version= "1.0" encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android"> 
< Scale android: duration = "5000" 
android:fromXScale ="1.0" 
android:fromYScale ="1.0" 
android: pivotX="50%" 
android:pivotY= "50%" 
android: toXScale = "0.0" 
android: toYScale = "0.0"/> 
</set > 


上 和 面 这 上段 代码 在 MyCode\MySample509\app\src\main\res\anim\myexit. xml 文件 中 。 此 实例 
的 完整 项 目 在 MyCode\MySample509 文件 夹 中 ， 


147 ”以 无 和 人 右 出 的 动画 效果 切换 两 个 Activity 


此 实例 主要 通过 使 用 overridePendingTransition() 方 法 ,实现 以 从 左 回 右 消 入 的 动画 效果 切换 两 
个 Activity。 当 实例 运行 之 后 ,图 147. 1 左 图 所 示 的 图 像 (MainActivity 的 背景 图 像 ) 将 回 右 滑 出 ， 
图 147. 1 右 图 所 示 的 图 像 (SecondActivity 的 背景 图 像 ) 将 从 左 滑 入 。 


MySample 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreatel( savedInstanceState) ; 
setContentView(R. layout. activity main); 
new Handler( ) .postDelayed(new Runnable( ){ 
(Override 
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public void run() { 
Intent myIntent = new Intent(MainActivity.this,SecondActivity.class); 
MainActivity. this. startActivity(myIntent); 
MainActivity. this. finish( ); 
overridePendingTransition(android.R.anim. slide in left, 
android. R. anim. slide out right); } },1000); // 实 现 由 左 向 右 滑 入 的 动画 效果 
} } 


上 和 面 这 段 代 码 在 MyCode\MySample297\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代码 中 ,overridePendingTransition() 方 法 用 于 在 参数 中 加 载 滑 入 和 滑 出 两 个 动 
国 | ,该 方法 通常 在 startActivity() 或 者 finishO 〇 执行 之 后 调用 。overridePendingTransition() 方 法 的 语 
法 声明 如 下 : 


overridePendingTransition(int enterAnim, int exitAnim) 


其 中 ,参数 int enterAnim 表示 跳 转 目标 Activity 的 进 场 动 画 。 参 数 int exitAnim 表示 当前 
Activity 的 离 场 动画 。 
此 实例 的 完整 项 目 在 MyCode\MySample297 文件 夹 中 ， 


148 ”以 收缩 扩张 的 动画 效果 切换 两 个 Activity 


此 实例 主要 通过 使 用 overridePendingTransition() 方 法 ,实现 以 收缩 扩张 的 动画 效果 切换 两 个 
Activity。 当 实例 运行 之 后 ,在 图 148. 1 左 图 所 示 的 图 像 中 ,飞机 所 代表 的 MainActivity 将 从 大 到 小 
逐渐 收缩 ,同时 轮船 所 代表 的 SecondActivity 将 从 小 到 大 逐渐 扩张 ,直到 完全 铺 满 屏幕 ,如 图 148. 1 
的 右 图 所 示 。 


MySample 


MySample 


图 148.1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
new Handler( ) .postDelayed(new Runnable() { 
(Override 
public void run() { 
Intent myIntent = new Intent(MainActivity.this, SecondActivity.class); 
MainActivity. this. startActivity(myIntent ); 
MainActivity. this. finish( ); 
overridePendingTransition(R. anim.myzoomin, R.anim. myzoomout ); 
} }, 500); // 以 收缩 和 扩张 动画 切换 两 个 Activity 
} } 


上 F 面 这 段 代 码 在 MyCode\ MySample299\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,overridePendingTransition() 方 法 用 于 在 参数 中 加 载 收 缩 
和 扩张 两 个 动画 ,扩张 动画 myzoomin 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android" > 
< Scale android: duration = "5000" 
android:fromXScale= "0.1" 
android:fromYScale = "0.1" 
android:pivotX= "50% p" 
android: pivotY = "50% p" 
android: toXScale = "1.0" 
android:toYScale = "1.0" /> 
</set> 


上 和 面 这 段 代码 在 MyCode\MySample299\app\src\main\res\anim\myzoomin. xml 文件 中 。 收 缩 
动画 myzoomonut 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< set xmlns:android = "http://schemas. android. com/apk/res/android"> 
< Scale android: duration = "5000" 
android:fromXScale = "1.0" 
android:fromYScale = "1.0" 
android: pivotX= "50%p" 
android: pivotY = "50% p" 
android:toXScale = "0.1" 
android:toYScale = "0.1" /> 


</set> 


上 面 这 段 代 码 在 MyCode\MySample299\app\src\main\res\anim\myzoomout. xml 文件 中 。 此 
实例 的 完整 项 目 在 MyCode\MySample299 文件 夹 中 。 
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149 ”使 用 转 场 动画 Explode 切换 两 个 Activity 


此 实例 主要 演示 了 在 startActivity() 方 法 中 设置 转 场 动画 ,实现 以 Explode 动画 方式 切换 两 个 
Activity。 当 实例 运行 之 后 , 单 击 “ 以 Explode 动画 进入 第 二 个 Activity” 按 钮 ,如 图 149. 1 的 左 图 所 
示 , 则 第 二 个 Activity 将 以 收缩 的 效果 出 现 。 在 第 二 个 Activity 中 单 击 “ 以 Explode 动画 返回 到 第 一 
个 Activity” 按 钮 ,如 图 149. 1 的 右 图 所 示 , 则 第 一 个 Activity 将 以 扩散 的 效果 出 现 。 


本 waQ 


MySample 


主要 代码 如 下 : 


// 响 应 单 击 "以 Explode 动画 进入 第 二 个 Rctivity" 按 钮 
public void onClickButtonl(View v) { 
Intent myIntent = new Intent(MainActivity.this,SecondActivity.class); 
startActivity(myIntent, 
ActivityOptions. makeSceneTransitionAnimation(this).toBundle( ) ) ; 


上 面 这 段 代 码 在 MyCode\MySample503\app\src\main\java\com\bin\luo\mysample\ MainAc- 
tivity. java 文件 中 。 在 这 段 代 码 中 ,startActivity(myIntent，ActivityOptions。makeSceneTransition- 
Animation(this). toBundle ()) 表 示 从 MainActivity 跳 转 到 SecondActivity 时 使 用 转 场 动画 。 


SecondActivity 的 主要 代码 如 下 : 


public class SecondActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity_ second) ; 
getWindow( ) . setEnterTransition(new Explode( ). setDuration(2000)); // 进 场 动 画 
getWindow( ) . setExitTransition(new Explode( ). setDuration(2000)); // 退 场 动画 
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} 
// 响 应 单 击 "以 Explode 动画 返回 到 第 一 个 Activity" 按钮 
public void onClickButton2(View v) { 
this. onBackPressed( ); 
} 
} 


上 面 这 上段 代码 在 MyCode\ MySample503\app\src\ main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 段 代 码 中 ,getWindow (). setEnterTransition (new Explode(). 
setDuration(2000)) 表示 在 进入 SecondActivity 时 ,使 用 转 场 动画 Explode, 持续 时 间 是 2 秒 。 
getWindow(). setExitTransition (new Explode(). setDuration(2000)) 表示 在 退出 SecondActivity 
时 ,使 用 转 场 动画 Explode, 持续 时 间 是 2 秒 。 此 外 , 当 在 项 目 中 新 增 了 SecondActivity, 则 需要 在 
AndroidManifest. xml 文件 中 注册 该 Activity ,如 < activity androlid:name 一 ". SecondActivity"/ >。 此 
实例 的 完整 项 目 在 MyCode\MySample503 文件 夹 中 。 


150 ”使 用 转 场 动画 Slide 切换 两 个 Activity 


此 实例 主要 通过 在 转 场 动画 Slide 的 setSlideEdge() 方 法 中 设置 滑动 标志 ,实现 以 定制 的 方向 切 
换 两 个 Activity。 当 实例 运行 之 后 , 单 击 “ 以 Slide 动画 进入 到 第 二 个 Activity” 按 钮 , 则 第 二 个 
Activity 将 从 屏幕 右边 滑 问 左边 至 全 屏 。 在 第 二 个 Activity 中 单 击 “以 Slide 动画 进入 到 第 一 个 
Activity” 按 钮 , 则 第 二 个 Activity 将 从 屏幕 左边 滑 回 右边 至 消失 ,从 而 显示 第 一 个 Activity, 效 果 分 别 
如 图 150. 1 的 左 图 和 右 图 所 示 。 
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以 Slidei ”以 Slide 动 画 进入 到 第 一 人 


// 响 应 单 击 "以 Slide 动画 进入 到 第 二 个 Activity" 按 钮 
public void onClickButtonl (View v) { 
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Intent myIntent = new Intent(MainActivity.this,SecondActivity.class); 
startActivity(myIntent, 
ActivityOptions. makeSceneTransitionAnimation(this).toBundle( )); 


上 和 面 这 上段 代码 在 MyCode\ MySample535 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 人 代码 中 ， startActivity (myIntent， ActivityOptions. 
makeSceneTransitionAnimation(this). toBundle()) 表 示 从 MainActivity 跳 转 到 SecondActivity 时 使 
用 转 场 动画 。SecondActivity 的 主要 代码 如 下 : 


public class SecondActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity_ second ) ; 
Slide mySlide = new Slide(); 


mySlide. setDuration(2000 ) ; 

mySlide. setSlideEdge( Gravity. RIGHT); 

getWindow( ) . setEnterTransition(mySlide); // 进 场 动画 
getWindow( ) . setExitTransition(mySlide); // 退 场 动画 
} 


// 响 应 单 击 "以 Slide 动画 进入 到 第 一 个 Rctivity" 按 钮 
public void onClickButton2(View v) { this. onBackPressed(); } 


} 


上 面 这 上段 代码 在 MyCode\ MySample535\app\src\ main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 段 代 码 中 ,mySlide. setSlideEdge(Gravity. 人 RIGHT) 表示 转 场 动 男 
的 滑动 方 回 是 从 右 回 左 。Gravity 的 成 员 较 多 ,setSlideEdge() 方 法 仅 支持 下 列 成 员 : Gravity. LEFT、 
Gravity. TOP .Gravity. RIGHT Gravity. BOTTOM Gravity. START .Gravity. END。 此 外 , 当 在 项 
目 中 新 增 了 SecondActivity, 则 需要 在 AndroidManifest. xml 文件 中 注册 该 Activity, 如 < activity 
android:name 一 ". SecondActivity "/ >。 此 实例 的 完整 项 目 在 MyCode\MySample535 文件 夹 中 。 


151 以 指定 位 置 的 转 场 动画 切换 两 个 Activity 


此 实例 主要 通过 使 用 ActivityOptionsCompat 的 makeScaleUpAnimation() 方 法 ,实现 以 指定 位 
置 和 大 小 的 放大 动画 来 切换 两 个 Activity。 当 实例 运行 之 后 ,在 MainActivity 中 单 击 任意 一 幅 图 像 
(如 左下 角 的 那 幅 图 像 ) , 则 将 以 该 图 像 的 位 置 和 大 小 放大 该 图 像 (SecondActivity) 至 全 屏 ,效果 分 别 
如 图 151. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


// 单 击 某 图 像 之 后 在 Secondactivity 中 显示 对 应 的 大 图 
public void onClickImageView(View v) { 
Intent myIntent = new Intent(this, SecondActivity.class); 
myIntent. putExtra(" MyImageID", v.getId()); 
ActivityOptionsCompat myActivityOptionsCompat = 
ActivityOptionsCompat. makeScaleUpAnimation(v, 0, 0, 
v.getMeasuredWidth(), v.getMeasuredHeight()); 
ActivityCompat. startActivity(MainActivity.this, myIntent, 
myActivityOptionsCompat. toBundle( ) ); 


人 > anaroia 红 应 用 200 。 实战 篇 


, 


上 \Mb,o 


. ~ - 
, a 上 A 4 


图 151.1 


上 面 这 段 代 码 在 MyCode\ MySample536\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myActivityOptionsCompat 一 ”ActivityOptionsCompat. 
makeScaleUpAnimation(v, 0，0,v. getMeasuredWidth(),，v. getMeasuredHeight()) 用 于 在 两 个 
Activity 切换 时 ,指定 执行 放大 动画 的 位 置 和 大 小 ,makeScaleUpAnimation() 方 法 的 语法 声明 如 下 : 


makeScaleUpAnimation(View source, int startxXx, 
int startY, int startWidth, int startHeight) 


其 中 ,参数 View source 是 一 个 view 对 象 , 用 于 确定 新 Activity 启动 的 初始 坐标 。 参 数 int 
startX 表示 新 Activity 出 现 的 初始 工 坐标 ,这 个 坐标 相对 于 source 的 左上 角 工 坐标。 参数 int startY 
表示 新 Activity 出 现 的 初始 y 坐标 ,这 个 坐标 相对 于 source 的 左上 角 > 坐标 。 参 数 int startWidth 表 
示 新 Activity 初始 的 宽度 。 参 数 startHeight 表示 新 Activity 初始 的 高 度 。 

myIntent 一 new Intent (this，SecondActivity. class) 表示 新 建 一 个 Intent, 该 Intent 实现 从 
MainActivity 切换 到 SecondActivity。SecondActivity 的 主要 代码 如 下 : 


public class SecondActivity extends Activity { 

ImageView myImageView; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
getWindow( ) . setFlags(WindowManager. LayoutParams. FLAG FULLSCREEN, 

WindowManager. LayoutParams. FLAG FULLSCREEN); // 设 置 全 屏 
setContentView(R. layout. activity_ second); 
int myImageID = getIntent().getExtras().getInt("MyImageID" ); 
myImageView = (ImageView)findViewById(R. id.myImageView); 
switch(myImageID) { 
case R. id.myViewl : 
myImageView. setImageResource(R. mipmap. myimage!l ); 
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break; 

case R. id. myView2: 
myImageView. setImageResource(R. mipmap. myimage2 ); 
break; 

case R. id. myView3: 


myImageView. setImageResource(R. mipmap. myimage3 ); 
break; 
case R. id. myView4: 
myImageView. setImageResource(R. mipmap. myimage4 ); 
break; 
} } 
// 单 击 大 图 像 之 后 返回 MainActivity 
public void onClickBigImage(View v) { this.onBackPressed( ); } 
} 


上 面 这 上段 代码 在 MyCode\ MySample536 \app\src\ main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 
'com. android. support:design:25. 0. 1 ' 依 赖 项 。 此 外 , 当 在 项 目 中 新 增 了 SecondActivity, 则 需要 在 
AndroidManifest. xml 文件 中 注册 该 Activity, 如 < activity android:name 王 “". SecondActivity"/ >。 此 
实例 的 完整 项 目 在 MyCode\MySample536 文件 夹 中 ， 


152 在 切换 Activity 时 答 加 绒 放 动画 和 转 场 动画 


此 实例 主要 演示 了 在 startActivity() 方 法 中 设置 转 场 动画 ,并 为 ImageView 控件 增加 缩放 动画 ， 
实现 在 从 MainActivity 跳 转 到 SecondActivity 时 同时 显示 转 场 动画 和 缩放 动画 。 当 实例 运行 之 后 ， 
MainActivity 将 显示 4 幅 电 影 海报 图 像 , 如 图 152. 1 的 左 图 所 示 ; 单 击 其 中 任意 一 幅 电 影 海报 图 像 ， 
如 右上 角 的 那 幅 图 像 , 则 SecondActivity 将 从 屏幕 顶部 滑 向 底部 ,同时 该 电影 海报 图 像 将 从 小 变 大 ， 
直到 填 满 整个 屏幕 ,如 图 152. 1 的 右 图 所 示 ; 单 击 后 退 按 钮 , 则 SecondActivity 将 从 屏幕 底部 滑 问 顶 
部 ,直到 消失 ,从 而 显示 第 一 个 Activity。 
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主要 代码 如 下 : 


// 单 击 某 图 像 之 后 在 SecondActivity 中 显示 对 应 的 大 图 
public void onClickmyBtnl (View v) { 
ActivityOptions myActivityOptions = 


ActivityOptions. makeSceneTransitionAnimation(this, v, ""); 


Intent myIntent = new Intent(this, SecondActivity.class); 
myIntent. putExtra(" MyImageID", v.getId()); 
startActivity(myIntent, myActivityOptions. toBundle( ) ); 

} 


上 面 这 段 代 码 在 MyCode\ MySample506\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ，startActivity(CmyIntent，myActivityOptions. toBundle()) 
表示 从 MainActivity 跳 转 到 SecondActivity 时 使 用 转 场 动画 。SecondActivity 的 主要 代码 如 下 : 


public class SecondActivity extends Activity { 
ImageView myImageView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
getWindow( ). setFlags(WindowManager. LayoutParams. FLAG FULLSCREEN, 
WindowManager. LayoutParams. FLAG FULLSCREEN); 
setContentView(R. layout. activity_second); 
getWindow( ) . setEnterTransition(new Explode( ). setDuration(2000)); 
getWindow( ) . setExitTransition(new Explode( ). setDuration(2000)); 
int myImageID = getIntent().getExtras().getInt("MyImageID" ); 
myImageView = (ImageView)findViewById(R. id. myImageView) ; 
Switch(myImageID){ 
case R. id. myViewl : 
myImageView. setImageResource(R. mipmap. myimagel ) ; 
break; 
case R. id. myYView2 : 
myImageView. setImageResource(R. mipmap.myimage2 ) ; 
break; 
case R. id. myView3: 
myImageView. setImageResource(R. mipmap. myimage3 ) ; 
break; 
case R. id. myView4: 
myImageView. setImageResource(R. mipmap. myimage4) ; 
break; 
} 
ScaleAnimation myScaleAnimation = new ScaleAnimation(0.0f, 1.0f, 
0.0f, 1.0f,Animation. RELATIVE TO SELF, 0.5f, 
Animation. RELATIVE TO SELF, 0.5f); 
myScaleAnimation. setDuration(2000); 
myImageView. setAnimation(myScaleAnimation); 
myScaleAnimation. start( ); 
}} 


上 面 这 上段 代码 在 MyCode\ MySample506 \app\src\ main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 段 代 码 中 ,getWindow (). setEnterTransition (new Explode(). 
setDuration(2000)) 表 示 在 进入 SecondActivity 时 ,使 用 转 场 动画 Explode ,持续 时 间 是 2 秒 ; 该 转 场 


// 设 置 全屏 


// 进 场 动 画 
// 退 场 动画 


// 创 建 缩放 动 画 
// 设 置 动 画 持续 时 间 2 秒 
// 在 ImageView 上 添加 缩放 动画 
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动画 将 与 ImageView 控件 的 缩放 动画 myScaleAnimation 同时 执行 。 getWindow ( ). 
setExitTransition(new Explode(). setDuration(2000)) 表 示 在 退出 SecondActivity 时 ,使 用 转 场 动画 
Explode, 持 续 时 间 是 2 秒 。 此 外 , 当 在 项 目 中 新 增 了 7 SecondActivity, 则 需要 在 AndroidManifest. xml 
文件 中 注册 该 Activity, 如 < activity android: name 王 ". SecondActivity"/ >>。 此 实例 的 完整 项 目 在 
MyCode\ MySample506 文件 夹 中 。 


153 ”在 切换 Activity 的 转 场 动画 中 共 诗 多 对 元 于 


此 实例 主要 通过 在 ActivityOptions 的 makeSceneTransitionAnimation() 方 法 参数 中 新 建 Pair 共 
人 享 元 素 键 值 对 ,从 而 实现 在 两 个 不 同 Activity 的 多 对 元 素 间 产生 转 场 动画 效果 。 当 实例 运行 之 后 ,由 
于 MainActivity 的 左右 两 个 ImageView 控件 (两 幅 电 影 海 报 ) 和 SecondActivity 的 上 下 两 个 
ImageView 控件 (两 幅 电影 海报 对 应 的 大 图 像 ) 被 设置 为 两 对 共享 元 素 , 因 此 在 单 击 MainActivity 的 
“进入 SecondActivity” 按 钮 之 后 ,在 SecondActivity 中 将 出 现在 MainActivity 左上 角 的 图 像 回 右 拉 
伸 ,在 MainActivity 右上 角 的 图 像 各 左下 角 拉 伸 的 效果 ,效果 分 别 如 图 153. 1 的 左 图 和 右 图 所 示 。 
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进入 SecondActivity 


主要 代码 如 下 : 


public void onClickmyBtnl(View v) { // 响 应 单 击 " 进 入 SecondActivity" 按 钮 
Intent myIntent = new Intent(this, SecondActivity. class) ; 
myIntent. putExtra("MyImagel", R.nipmap.myimagel ) ; 
myIntent. putExtra("MyImage2", R.mipmap. myimage2); 
ActivityOptions options = ActivityOptions.makeSceneTransitionAnimation 
(MainActivity.this,new Pair <View,String >(myImageViewl,"mySharel" ), 
new Pair < View, String >(myImageView2,"myShare2" ) ) ; 
startActivity(myIntent, options. toBundle( ) ) ; 


上 面 这 上段 代码 在 MyCode\ MySample537\app\src\ main\java\com\bin\luo\mysample\ 
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MainActivity. java 文件 中 。 在 这 段 代 码 中 ,Pair < View,String >(myImageView1l，"mySharel" ) 与 
SecondActivity 中 的 ViewCompat. setTransitionName(myImageViewA, "mySharel") 形 成 一 对 共享 
元 素 ; Pair < View, String > (myImageView2," myShare2") 与 SecondActivity 中 的 ViewCompat. 
setTransitionName(mylmageViewB, "myShare2") 形 成 一 对 共享 元 素 。SecondActivity 的 主要 代码 
如 下 : 


public class SecondActivity extends Activity { 

ImageView myImageViewA, mylImageViewB; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
getWindow( ). setFlags(WindowManager. LayoutParams. FLAG FULLSCREEN, 

WindowManager. LayoutParams. FLAG FULLSCREEN); 

setContentView(R. layout.activity_ second); 
myImageViewA = (ImageView)findViewById(R. id. myImageViewR) ; 
myImageViewB = (ImageView)findViewById(R. id. myImageViewB) ; 
ViewCompat. setTransitionName(myImageViewA, "mySharel" ) ; 
ViewCompat. setTransitionName(myImageViewB, "myShare2" ); 
int myID1 = getIntent().getExtras().getInt("MyImagel"); 
int myID2 = getIntent().getExtras().getInt("MyImage2"); 
myImageViewA. setImageResource( myID!] ); 
myImageViewB. setImageResource(myID2 ) ; 

} 

// 单 击 大 图 像 之 后 返回 MainActivity 

public void onClickBigImage(View v) { this. onBackPressed( ); } 

} 


上 面 这 段 代 码 在 MyCode\ MySample537\app\src\ main\java\com\bin\luo\ mysample\ 
SecondActivity. java 文件 中 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 
'com. android. support:design:25. 0.1' 依 赖 项 。 此 外 , 当 在 项 目 中 新 增 了 SecondActivity, 则 需要 在 
AndroidManifest. xml 文件 中 注册 该 Activity ,如 < activity android:name 王 “. SecondActivity"/ >。 此 
实例 的 完整 项 目 在 MyCode\MySample537 文件 夹 中 ， 


154 ”使 用 FragmentTransaction 自 定 义 转 场 动画 


此 实例 主要 通过 使 用 FragmentTransaction 的 setCustomAnimations ( ) 方法 ,实现 在 切换 
Fragment 时 显示 自 定 义 的 转 场 动画 。 当 实例 运行 之 后 ,将 显示 默认 的 Fragment" 异形 魔 怪 ?; 单 击 第 
二 个 Fragment 的 标签 “头脑 特工 队 ”, 则 Fragment 异形 魔 怪 ? 将 回 右 滑 出 ,Fragment 头脑 特工 队 ? 将 
从 左 滑 入 ,如 图 154.1 的 左 图 所 示 ; 单 击 第 三 个 Fragment 的 标签 “赛车 总 动员 ”, 则 Fragment 头脑 特 
工 队 ”将 回 右 滑 出 ,Fragment 赛 车 总 动员 ?将 从 左 滑 入 ,如 图 154. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends FragmentActivity 
implements View. OnClickListener{ 
private RadioButton myRadioButtonl ; 
private RadioButton myRadioButton2; 
private RadioButton myRadioButton3; 
private RadioGroup myRadioGroup; 


Private android. support. v4. app. Fragment myFragmentl1, myFragment2, myFragment3; 
private android. support. v4. app. FragmentManager myFragmentManager; 
private android. support. v4. app. FragmentTransaction myFragmentTransaction; 
(QOverride 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
requestWindowFeature(Window. FEATURE NO_TITLE); 
setContentView(R. layout. activity main); 
myFragmentManager = getSupportFragmentManager( ); 
myFragmentTransaction = myFragmentManager. beginTransaction( ) ; 
myRadioButtonl = (RadioButton) findViewById(R. id. myYRadioButtonl ) ; 
myRadioButton2 (RadioButton) findViewById(R. id. myRadioButton2 ) ; 
myRadioButton3 = (RadioButton) findViewById(R. id. myRadioButton3 ) ; 
myRadioGroup = (RadioGroup) findViewById(R. id. rg); 
// 为 三 个 按钮 (标签 ) 添 加 监听 
myRadioButton1l. setOnClickListener(this); 
myRadioButton2. setOnClickListener(this); 
myRadioButton3. setOnClickListener(this); 
// 启 动 默 认 选 中 第 一 个 
myRadioGroup. check(R. id. myYRadioButtonl ) ; 
myFragmentl = new Fragmentl( ) ; 


myFragmentTransaction. replace(R. id. myContent, myFragment] ) ; 
myFragmentTransaction. commit( ) ; 
} 
(Override 
public void onClick(View v) { 
myFragmentManager = getSupportFragmentManager( ); 
myFragmentTransaction = myFragmentManager. beginTransaction( ) ; 
myFragmentTransaction. setCustomAnimations(android.R.anim. slide in left, 
android.R. anim. slide out right); // 添 加 转 场 动画 
Switch (v.getId()) { 
case R. id. myYRadioButtonl : 
myFragmentl1l = new Fragmentl( ) ; 
myFragmentTransaction. replace(R. id. myContent, myFragment] ) ; 
break; 
case R. id. myRadioButton2: 
myFragment2 = new Fragment2(); 
myFragmentTransaction. replace(R. id. myContent, myFragment2); 
break; 
case R. id. myRadioButton3: 
myFragment3 = new Fragment3(); 
myFragmentTransaction. replace(R. id. myContent, myFragment3); 
break; 
default: 
break; 
} 
myFragmentTransaction. commit( ) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySample513\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myFragmentTransaction。setCustomAnimations 
(android. R. anim. slide_in_left,android. R. anim. slide_out_right) 用 于 自 定义 三 个 Fragment 的 转 场 
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图 154. 1 


动画 ,android.R. anim. slide_ in_left 表示 进 场 动画 ,android. R. anim. slide_out_right 表示 退场 动画 。 
需要 说 明 的 是 ,在 Android Studio 中 使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. 
support:support-v4:24. 2.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample513 文件 夹 中 。 


155 ”使 用 TransitionManasger 实现 上 下 请 动 动画 


此 实例 主要 通过 指定 TransitionManager 的 beginDelayedTransition() 方 法 参数 为 容 占 (控件 的 
父 节 点 ) ,然后 在 修改 布局 参数 后 调用 setLayoutParams() 方 法 ,从 而 实现 在 修改 布局 参数 后 的 子 节 点 
(控件 ) 产 生 位 移 效果 的 过 渡 动 画 。 当 实例 运行 之 后 ,小 鸡 图 像 (ImageView 控件 ) 位 于 屏幕 顶端 , 单 击 
屏幕 , 则 小 鸡 图 像 将 沿 着 箭头 方 回 从 屏幕 顶部 滑 回 底部 ,如 图 155. 1 的 左 图 所 示 。 再 次 单 击 屏幕 , 则 
小 鸡 图 像 沿 着 篆 头 方 回 从 屏幕 底部 滑 回 顶部 ,如 图 155. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

RelativeLayout myLayout; 

boolean bChanged = false; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myLayout = (RelativeLayout)findViewById(R. id. myLayout); 

} 

// 单 击 屏 幕 实现 ALIGN_PRRENT TOP 和 ALIGN _PRARENT BOTTOM 过 渡 动 画 切 换 

public void onClickScreen(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myRootView) ; 
TransitionManager. beginDelayedTransition(myRootView); 
RelativeLayout. LayoutParams lp = new RelativeLayout. LayoutParams( 

myLayout. getMeasuredWidth( ),myLayout. getMeasuredHeight( )); 
if (bChanged) { lp.addRule(RelativeLayout. ALIGN PARENT TOP); } 
else { lp.addRule(RelativeLayout. ALIGN PARENT BOTTOM); } 
bChanged = ! bChanged; 
myLayout. setLayoutParams( 1p); 
}} 


上 面 这 段 代 码 在 MyCode\ MySample592\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 ，TransitionManager。 beginDelayedTransition 
(myRootView) 表示 在 父 容 磊 myRootView 中 进行 过 渡 动 画 。myRootView 一 (ViewGroup ) 
findViewByIdCR. id. myRootView) 用 于 根据 activity_main 布局 中 的 RelativeLayout 创建 动画 容 骨 。 
activity_main 布局 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 
< RelativeLayout xmlns:android = "http://schemas.android. com/apk/res/android" 
xmlns:tools = "http://schemas. android. com/tools" 
android:id= "(@ + id/activity main" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:onClick = "onClickScreen" 
tools:context = "com. bin. luo. mysample. MainActivity"> 
<! 一 一 这 个 RelativeLayout 用 来 做 动画 的 父 布局 -一 > 
< RelativeLayout android: id = "(@ + id/myRootView" 
android:gravity = "center horizontal" 
android: layout_width = "match_ parent" 
android:layout height = "match parent"> 
< include layout = " (layout/myscenel"></ include > 
</RelativeLayout ></RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample592\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,< include layout 二" (@ layout/myscenel">></include > 表示 容器 中 的 子 布 局 是 
myscenel ,myscenel 布局 的 主要 内 容 如 下 : 


<?xml] version = "1.0" encoding = "utf 一 8"?> 

< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
android: id= "@ + id/myLayout" 
android:layout width= "wrap_content” 
android:layout height = "wrap_content"> 
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< ImageView android: layout width= "100dp" 
android:layout height = "100dp" 
android: src = "(mipmap/myimagel"/> 
</RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample592\app\src\main\res\layout\myscenel. xml 文件 中 。 需 
要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 
25.0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample592 文件 夹 中 ， 


156 ”使 用 TransitionManager 实现 围绕 YY 轴 诞 转 


此 实例 主要 通过 指定 TransitionManager 的 beginDelayedTransition() 方 法 参数 分 别 为 容 需 ( 控 
件 的 父 节 点 ) 和 ChangeTransform 动画 ,并 修改 控件 的 RotationY 属性 ,从 而 实现 控件 围绕 y 轴 旋 转 
的 动画 效果 。 当 实例 运行 之 后 , ImageView 控件 (电影 海报 图 像 ) 位 于 屏幕 中 心 , 单 击 屏幕 , 骨 
ImageView 控件 (电影 海报 图 像 ) 将 围绕 y 轴 正 向 旋转 45*, 如 图 156. 1 的 右 图 所 示 。 再 次 单 击 屏幕 ， 
则 ImageView 控件 (电影 海报 图 像 ) 将 围绕 y 轴 反 回旋 转 45 ,如 图 156. 1 的 左 图 所 示 。 


MySample MySample 


主要 代码 如 下 : 


// 单 击 屏幕 实现 图 像 围 绕 y 轴 旋 转 的 动画 切换 
public void onClickScreen(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myYRootView) ; 
TransitionManager. beginDelayedTransition(myRootView, 
new ChangeTransform( ). setDuration(2000)); 
if (bChanged) { myImageView. setRotationY( 0f);)} 
else { myImageView. setRotationY(45f); } 
bChanged = ! bChanged; 


上 F 面 这 段 代 码 在 MyCode\ MySample600\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 ，TransitionManager. beginDelayedTransition 
(myRootView,new ChangeTransform (). setDuration (2000)) 表 示 在 父 容器 myRootView 中 执行 
ChangeTransform 过 渡 动 画 ,ChangeTransform 过 渡 动 画 主要 用 于 改变 ImageView 控件 的 缩放 比例 
和 旋转 角度 。myRootView 二 (ViewGroup)findViewByld(R. id. myRootView) 用 于 根据 activity 
main 布局 中 的 RelativeLayout 创建 动画 容器 。activity_main 布局 的 主要 内 容 如 下 : 


<?Xxml version= "1.0" encoding = "utf 一 8"?> 
< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns :tools = "http://schemas. android. com/tools" 
android:id= "@ + id/activity main" 
android:layout width = "match parent" 
android:layout height = "match parent" 
android:onClick = " onClickScreen" 
tools:context = "com. bin. luo. mysample. MainActivity"> 
<! 一 一 这 个 RelativeLayout 用 来 做 动画 的 父 布 局 -一 > 
< RelativeLayout android: id = "(@ + id/myRootView" 
android:gravity = "center_ vertical |center horizontal" 
android: layout _ width= "match parent" 
android:layout height = "match parent"> 
< ImageView android:layout width= "wrap_content" 
android:layout height = "wrap_content" 
android:padding = "50dp" 
android: src = "(@mipmap/myimagel" 
android: id= "(@ + id/myImageView" /> 
</RelativeLayout ></RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample600\app\src\main\res\layout\activity_main. xml 文件 中 。 
需要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 
25.0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample600 文件 夹 中 。 


157 使 用 TransitionManager 实现 Fade 动画 效果 


此 实例 主要 通过 指定 TransitionManager 的 beginDelayedTransition() 方 法 的 参数 分 别 为 容器 
(控件 的 父 节 点 ) 和 Fade 动画 ,然后 通过 日 定义 方法 改变 控件 的 Visibility 属性 ,从 而 实现 控件 在 显示 
或 隐藏 时 产生 淡 入 淡出 的 过 渡 动 画 效 果 。 当 实例 运行 之 后 ,两 个 ImageView 控件 (电影 海报 图 像 ) 位 
于 屏幕 中 心 ,如 图 157. 1 的 左 图 所 示 ; 单 击 屏幕 , 则 当前 的 两 个 ImageView 控件 (电影 海报 图 像 ) 淡 
出 ,显示 另外 两 个 ImageView 控件 (电影 海报 图 像 ) ,如 图 157. 1 的 右 图 所 示 。 再 次 单 击 屏幕 , 则 以 前 
隐藏 的 两 个 ImageView 控件 (电影 海报 图 像 ) 淡 入 ,如 图 157. 1 的 左 图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
ImageView myImageViewl],myImageView2,myImageView3, myImageView4; 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myImageViewl = (ImageView)findViewById(R. id.myImageView!] ) ; 
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myImageView2 = (ImageView)findViewById(R. id. myImageView2 ) ; 
myImageView3 = ( ImageView)findViewBVyId(R. id. myImageView3 ) ; 
myImageView4 = (ImageView)findViewById(R. id. myImageView4) ; 


} 
// 单 击 屏幕 实现 淡 和 人 淡出 动画 切换 
public void onClickScreen(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myYRootView) ; 
TransitionManager. beginDelayedTransition( myRootView, 
new Fade( ). setDuration(5000)); 
toggleVisibility(myImageView3, myImageView4); 
} 
private static void toggleVisibility(View... views){ 
for (View v:views) { 
boolean isVisible = v.getVisibility() == View.VISIBLE; 
V. setVisibility(isVisible?View. INVISIBLE: View. VISIBLE); 
}y]} 
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图 157.1 


上 和 面 这 段 代码 在 MyCode\MySample594\app\src\main\java\com\bin\luo\mysample\MainActivity. 
java 文件 中 。 在 这 段 代 人 码 中 ,TransitionManager. beginDelayedTransition (myRootView, new Fade(). 
setDuration (5000 )) 表示 在 父 容 天 myRootView 中 执行 淡出 淡 入 过 渡 动 男 。ViewGroup 
myRootView 王 (ViewGroup) findViewById(CR. id. myRootView) 用 于 根据 activity_main 布局 中 的 


RelativeLayout 创建 动 夯 容器 。activity_main 布局 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 

< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns:tools = "http://schemas.android. com/tools" 
android: id = "(@ + id/activity main" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android: onClick = "onClickScreen" 
tools:context = "com. bin. luo. mysample. MainActivity"> 

<! 一 一 这 个 RelativeLayout 用 来 做 动画 的 父 布局 -一 > 


< RelativeLayout android: id= "(@O + id/myRootView" 
android:gravity = "center vertical" 
android:layout width= "match parent" 
android:layout height = "match parent"> 
< include layout = "(Q layout/myscenel"></ include > 
</RelativeLayout ></RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample594\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,< include layout 三 "(@ layout/myscenel"> </include > 表示 容器 中 的 子 布 局 是 
myscenel ,myscenel 布局 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
android: id= "(@ + id/myLayout" 
android:layout width = "wrap_content" 
android:layout height = "wrap_content"> 
<LinearLayout android:layout width= "match parent" 
android:layout height = "wrap content" 
android:orientation = "horizontal"> 
< ImageView android: id= "(@ + id/myImageViewl" 
android:layout width = "match parent" 
android:layout height = "match parent" 
android: layout margin = "2dp" 
android:layout weight = "1" 
android: scaleType = "centerCrop" 
android: src = "mipmap/myimagel" /> 
< ImageView android: id = "(@ + id/myImageView2" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:layout margin = "2dp" 
android:layout weight = "1" 
android: scaleType = "centerCrop" 
android:src = "(@mipmap/myimage2"/> 
</LinearLayout > 
<LinearLayout android:layout width= "match parent" 
android: layout height = "wrap_content" 
android:orientation = "horizontal"> 
< ImageView android: id = "(@ + id/myImageView3" 
android:layout width = "match parent" 
android:layout_ height = "match parent" 
android:layout margin = "2dp" 
android:layout weight = "1" 
android: scaleType = "centerCrop" 
android:src = "(Wmipmap/myimage3"/> 
< ImageView android: id= "(@ + id/myImageView4" 
android:layout width = "match parent" 
android:layout height = "match parent" 
android:layout margin = "2dp" 
android:layout weight = "1" 
android: scaleType = "centerCrop" 
android:src = "(Omipmap/myimage4" /> 
</LinearLayout > 
</RelativeLayout > 


上 和 面 这 段 代码 在 MyCode\MySample594\app\src\main\res\layout\myscenel. xml 文件 中 。 需 
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要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android.support:design:25. 0. 1' 依 
赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample594 文件 夹 中 。 


158 ”使 用 TransitionManager 组 合 多 个 不 同 动 夯 


此 实例 主要 通过 在 TransitionManager 的 beginDelayedTransition () 方法 中 组 合 Explode 和 
ChangeBounds 动画 ,实现 放大 缩小 图 像 和 四 散 飞 出 的 特殊 效果 。 当 实例 运行 之 后 ,四 幅 图 像 并 列 居 
中 ,如 图 158. 1 的 左 图 所 示 。 单 击 任意 一 幅 图 像 , 如 右上 角 的 图 像 , 则 该 图 像 放 大 2.5 倍 , 其 他 图 像 沿 
着 各 上 自 的 方 回 飞 出 屏幕 ,如 图 158. 1 的 右 图 所 示 。 单 击 放 大 的 图 像 , 则 该 图 像 恢 复原 状 ,其 他 图 像 沿 
着 各 目的 方向 飞 入 屏幕 。 


pm 
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图 158.1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements View. OnClickListener{ 
private ImageView myImageViewl,myImageView2, myImageView3, mylImageView4; 
private boolean bLarge; 
private ViewGroup myRootView; 
private int mySize; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myRootView = (ViewGroup) findViewById(R. id. myRootView); 
myImageViewl = (ImageView) findViewById(R. id. myImageViewl ) ; 
myImageView2 = (ImageView) findViewById(R. id. myImageView2 ) ; 
myImageView3 = (ImageView) findViewById(R. id. myImageView3 ) ; 
myImageView4 = (ImageView) findViewById(R. id. myImageView4 ) ; 
mySize = myImageView1l. getLayoutParams( ) . width; 
myImageView1. setOnClickListener(this); 


myImageView2. setOnClickListener(this); 
myImageView3. setOnClickListener(this); 
myImageView4. setOnClickListener(this); 
} 
// 响 应 单 击 当 前 的 ImageView 控件 
public void onClick(View view) { 
Explode myExplode = new Explode( ); 
ChangeBounds myChangeBounds = new ChangeBounds( ); 
TransitionSet myTransitions = new TransitionSet( ); 
myTransitions. addTransition(myExplode).addTransition(myChangeBounds); 
myTransitions. setDuration( 500); 
TransitionManager. beginDelayedTransition(myRootView, myTransitions); 
changeSize( view); 
changeVisibility(myImageViewl, myImageView2, mylImageView3, myImageView4); 
View. setVisibility(View. VISIBLE); 
} 
// 放 大 当前 单 击 的 ImageView 控件 (导致 ChangeBounds 动画 发 生 ) 
private void changeSize(View view) { 
bLarge = !bLarge; 
ViewGroup. LayoutParams layoutParams = view.getLayoutParams( ) ; 
if(bLarge){ 
layoutParams.width = (int)(2.5*x* mySize); 
layoutParams. height = (int)(2.5*x mySize); 
}else { 
layoutParams.width = mySize; 
layoutParams. height = mySize; 
} 
view. setLayoutParams( layoutParams); 
} 
// 隐 藏 (或 显示 ) 非 当前 单 击 的 ImageView 控件 (导致 Explode 动画 发 生 ) 
private void changeVisibility(View . . .views){ 
for (View view:views){view. setVisibility(view. getVisibility() == 
View. VISIBLE?View. INVISIBLE :View. VISIBLE); } 
上 


上 面 这 段 代 码 在 MyCode\ MySample628\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 ，TransitionManager. beginDelayedTransition 
(myRootView，myTransitions) 中 的 myTransitions 动画 在 控件 尺寸 发 生 改 变 或 Visibility 属性 发 生 
变化 时 触发 。myRootView = 三 (ViewGroup) findViewById(CR. id. myRootView) 用 于 根据 activity 
main 布局 中 的 RelativeLayout 创建 动画 容器 。activity_main 布局 的 主要 内 容 如 下 : 


< RelativeLayout android: id = "(@ + id/myRootView" 
android:layout width= "match parent” 
android:layout height = "match parent" 
android:layout centerInParent = "true" 
android:gravity = "center horizontal |center vertical"> 
< ImageView android: id= "(@ + id/myImageViewl" 
android: layout width = "100dp" 
android: layout height = "100dp" 
android:layout margin = "10dp" 
android: src = "(@mipmap/myimagel"/> 
< ImageView android: id = "(@ + id/myImageView2" 
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< ImageView 


< ImageView 
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android: 
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</RelativeLayout > 


layout width= "100dp" 

layout height = "100dp" 

layout margin = "10dp" 

layout toRightOf = "(@ id/myImageView1" 
src= "(mipmap/myimage2"/> 

id= "(@ + id/myImageView3" 

layout width= "100dp" 

layout height = "100dp" 

layout below = "(@ id/myImageView1" 
layout margin = "10dp" 

src= "(mipmap/myimage3"/> 

id= "(@ + id/myImageView4" 

layout width= "100dp" 

layout height = "100dp" 

layout below = "(@ id/myImageViewl1" 
layout margin = "10dp" 

layout toRightOf = "(@ id/myImageView3" 
src= "(Qnmipmap/myimage4" /> 


上 面 这 段 代 码 在 MyCode\MySample628\app\src\main\res\layout\activity_main. xml 文件 中 。 
此 实例 的 完整 项 目 在 MyCode\MySample628 文件 夹 中 。 


159 ”使 用 TransitionManager 实现 单 布 局 过 渡 动 画 


此 实例 主要 通过 使 用 setLayoutParams() 方 法 修改 控件 的 LayoutParams 布局 参数 ,实现 使 
TransitionManager 的 beginDelayedTransition() 方 法 中 的 ChangeBounds 过 渡 动 画 参 数 ( 图 像 ) 产 生 
动画 效果 。 当 实例 运行 之 后 ,小 鸟 图 像 (ImageView 控件 ) 位 于 屏幕 左上 和 角 , 单 击 屏幕 , 则 小 乌 图 像 将 
治 着 弧 线 路 径 ( 细 实 线 , 实 际 不 可 见 ) 从 屏幕 左上 角 滑 回 右 下 角 , 如 图 159. 1 的 左 图 所 示 。 再 次 单 击 屏 
幕 , 则 小 乌 图 像 将 沿 弧 线路 径 从 屏幕 右 下 角 滑 向 屏幕 左上 角 ,如 图 159. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


// 单 击 屏 幕 实现 左上 角 和 右 下 角 的 弧 线 路 径 动 画 切 换 
public void onClickScreen(View v) { 
ChangeBounds myChangeBounds = new ChangeBounds( ); 
myChangeBounds. setPathMotion(new ArcMotion( ) ); // 设 置 弧 线 路 径 
myChangeBounds. setDuration(5000); // 设 置 动画 时 间 5 秒 
ViewGroup myRootView = (ViewGroup) findViewById(R. id.myRootView); 
TransitionManager. beginDelayedTransition(myRootView, myChangeBounds); 
FrameLayout. LayoutParams myLayoutParams = 
(FrameLayout. LayoutParams) myImageView. getLayoutParams( ); 
myLayoutParams.gravity = bChanged ? (Gravity.LEFT | Gravity.TOP) : 
(Gravity. BOTTOM | Gravity. RIGHT); 
myImageView. setLayoutParams(myLayoutParams); 
bChanged = !bChanged; 


上 面 这 段 代 码 在 MyCode\ MySample610\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 人 代码 中 ， TransitionManager. beginDelayedTransition 
(myRootView，myChangeBounds) 表 示 在 父 容 胡 myRootView 中 执行 myChangeBounds 动画 。 在 
Android 中 ,ChangeBounds 动画 通常 在 LayoutParams 属性 发 生 改 变 的 情况 下 执行 。 
myLayoutParams. gravity = bChanged ? (Gravity. LEFT | Gravity. TOP) : (Gravity. BOTTOM | 
Gravity. RIGHT) 和 myImageView. setLayoutParams (myLayoutParams) 表 示 如 果 当 前 ImageView 
控件 在 屏幕 左上 角 , 则 通过 设置 LayoutParams 属性 将 其 置 于 右 下 角 ; 如 果 当 前 ImageView 控件 在 屏 
幕 右 下 角 , 则 通过 设置 LayoutParams 属性 将 其 置 于 左上 角 ; 一 旦 控件 的 LayoutParams 属性 发 生 改 
变 即 触发 ChangeBounds 动画 。myRootView 二 (ViewGroup) findViewByld(R. id. myRootView ) 用 
于 根据 activity_main 布局 中 的 FrameLayout 创建 动画 容器 。activity_main 布局 的 主要 内 容 如 下 : 


< FrameLayout android: id = "(@ + id/myRootView" 
android: layout width = "match parent" 
android:layout height = "match parent"> 

< ImageView android:layout width= "150dp" 

android:layout height = "200dp" 
android:padding = "2dp" 
android: layout alignParentLeft = "true" 
android: layout_alignParentTop = "true" 
android: src = "(mipmap/myimagel" 
android: id= "(@ + id/myImageView" /> 

</FrameLayout > 


上 面 这 段 代 码 在 MyCode\MySample610\app\src\main\res\layout\activity_main. xml 文件 中 。 
此 实例 的 完整 项 目 在 MyCode\MySample610 文件 夹 中 。 


160 ”使 用 TransitionManasger 实现 平移 过 渡 动 画 


此 实例 主要 通过 使 用 TransitionManager 的 go() 方 法 ,使 两 个 布局 (场景 ) 实 现 以 Slide 动画 风格 
滑 出 滑 入 。 当 实例 运行 之 后 ,将 全 屏 显 示 重 庆 图 像 (布局 ) , 单 击 重庆 图 像 , 则 该 图 像 将 回 下 滑 出 ,同时 
上 海 图 像 ( 布 局 ) 则 回 上 滑 入 至 全 屏 显 示 ,效果 分 别 如 图 160. 1 的 左 图 和 右 图 所 示 。 
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Q 卫 加 中 四 和 dD "dM 100% 下 午 6:39 


主要 代码 如 下 : 


// 单 击 重庆 图 像 ( 布 局 ) 之 后 过 渡 到 上 海 图 像 ( 布 局 ) 
public void onClickImageViewl (View v) { 
ViewGroup myMainLayout = (ViewGroup)findViewById(R. id.activity main); 
Scene mySecondScene = Scene.getSceneForLayout(myMainLayout, 
R. layout. layout second, this); 
Slide mySlide = new Slide(); 
mySlide. setDuration( 5000); 
mySlide. setSlideEdge( Gravity. BOTTOM); 
Transition myTransition = mySlide; 
TransitionManager. go(mySecondScene, myTransition); 
} 
// 单 击 上 海 图 像 (布局 ) 之 后 过 渡 到 重庆 图 像 ( 布 局 ) 
public void onClickImageView2(View v) { 
ViewGroup mySecondLayout = (ViewGroup)findViewById(R. id. layout second); 
Scene myMainScene = Scene.getSceneForLayout(mySecondLayout, 
R. layout.activity main, this); 
Slide mySlide = new Slide( ); 
mySlide. setDuration( 5000); 
mySlide. setSlideEdgel( Gravity. BOTTOM); 
Transition myTransition = mySlide; 
TransitionManager. go(myMainScene, myTransition); 


上 面 这 段 代 码 在 MyCode\ MySample544\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySlide. setSlideEdge(Gravity. BOTTOM) 表 示 当 前 布局 
(图 像 ) 从 底部 滑 出 、 其 他 图 像 从 底部 滑 入 ; 如 果 是 mySlide. setSlideEdge (Gravity. TOP) , 则 表示 当 
前 布局 (图 像 ) 从 顶部 滑 出 、 其 他 图 像 从 顶部 滑 入 ; 如 果 是 mySlide. setSlideEdge(Gravity. LEFT), 则 
表示 当前 布局 (图 像 ) 从 左边 滑 出 、 其 他 图 像 从 左边 滑 入 ; 如 果 是 mySlide. setSlideEdge (Gravity. 
RIGHT), 则 表示 当前 布局 (图 像 ) 从 右边 滑 出 、 其 他 图 像 从 右边 滑 入 。TransitionManager. go 
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(mySecondScene，myTransition) 表 示 从 activity main 布局 过 渡 到 layout_second 布局 。 关 于 这 两 个 
布局 文件 的 主要 代码 请 参考 源 代码 中 的 MyCode\MySample544\app\src\main\res\layout\activity_ 
main. xml 文件 和 MyCode\ MySample544\app\src\main\res\layout\layout_second. xml 文件 。 需 要 
说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support:design: 25. 0. 1， 
依赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample544 文件 夹 中 。 


161 使 用 TransitionManager 实现 缩放 部 分 图 像 


此 实例 主要 通过 使 用 setClipBounds() 方 法 指定 场景 过 渡 动 画 的 (裁剪 ) 范 围 ,实现 在 两 个 布局 过 
渡 的 过 程 中 仅 放大 或 缩小 控件 (ImageView) 在 范围 内 的 部 分 (图 像 )。 当 实例 运行 之 后 ,ImageView 
控件 (图 像 ) 位 于 屏幕 的 正中 ( 即 myscenel ) , 单 击 屏 幕 , 则 ImageView 控件 (图 像 的 右 下 角 部 分 ) 将 由 
小 变 大 ,直到 铺 满 屏幕 的 右 下 角 ,如 图 161. 1 的 右 图 所 示 ( 即 myscene2) 。 在 myscene2 中 单 击 屏幕 , 则 
ImageView 控件 (图 像 的 右 下 角 部 分 ) 将 由 大 变 小 ,如 图 161. 1 的 左 图 所 示 ( 即 myscenel ) 。 
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图 161. 1 
主要 代码 如 下 : 


// 单 击 myscenel 过 渡 到 myscene2( 图 像 的 右 下 角 由 小 变 大 ) 
public void onClickScenel(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myYRootView) ; 
// 设 置 过 渡 动 画 的 范围 在 右 下 角 
myRootView. setClipBounds(new Rect (myRootView. getWidth( )/2, 
myRootView. getHeight()/2,myRootView. getWidth( ) ,myRootView. getHeight( ))); 
Scene myScene2 = Scene.getSceneForLayout(myRootView, R. layout.myscene2, this); 
TransitionManager. go(myScene2, new ChangeBounds( ) . setDuration(5000)); 
} 
// 单 击 myscene2 过 渡 到 myscenel (图像 的 右 下 角 由 大 变 小 ) 
public void onClickScene2(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
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// 设 置 过 渡 动 画 的 范围 在 右 下 角 
myRootView. setClipBounds(new Rect(myRootView. getWidth( )/2, 

myRootView. getHeight()/2,myRootView. getWidth(),myRootView. getHeight())); 
Scene myScenel = Scene.getSceneForLayout(myRootView, R. layout.myscenel, this); 
TransitionManager.go(myScenel, new ChangeBounds( ). setDuration(5000)); 


上 面 这 上段 代码 在 MyCode\ MySample554 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,TransitionManager. go(myScene2 ，new ChangeBounds 
(). setDuration(5000)) 表 示 以 ChangeBounds 动画 风格 在 5 秒 内 从 myScenel 过 渡 到 myScene2 , 即 图 
像 的 在 下 角 由 小 变 大 。TransitionManager。 go (myScenel, new ChangeBounds ( ). setDuration 
(5000)) 表示 以 ChangeBounds 动画 风格 在 5 秒 内 从 myScene2 过 渡 到 myScenel , 即 图 像 的 右 下 角 由 
大 变 小 。ViewGroup myRootView 二 (ViewGroup ) findViewById (R. id. myRootView ) 用 于 根据 
activity_main 布局 中 的 RelativeLayout 创建 动画 容器 。myRootView. setClipBounds (new Rect 
(myRootView. getWidth ( )/2, myRootView. getHeight ( )/2, myRootView。 getWidth ( )， 
myRootView. getHeight())) 表 示 TransitionManager. go() 方 法 在 执行 过 渡 动 画 时 , 仅 在 裁剪 的 Rect 
中 执行 ( 即 图 像 的 在下 角 )。 关 于 activity_main myscenel .myscene2 三 个 布局 的 详细 内 容 请 参考 源 
代码 中 的 MyCode\MySample554\app\src\main\res\layout\activity_main. xml 文件 .MyCode'\ 
MySample554\app\src\main\res\layout\myscenel. xml 文件 和 MyCode\MySample554\ app\src\ 
main\res\layout\myscene2. xml 文件 。 此 外 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. 
android. support: design:25. 0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample554 文件 夹 中 。 


162 ”使 用 TransitionManager 实现 矢量 路 径 动 画 


此 实例 主要 通过 在 TransitionManager 的 go() 方 法 中 加 载 在 XML 文件 中 定制 的 过 渡 动 画 ,实现 
在 两 个 布局 (myscenel 和 myscene2) 切 换 时 ,ImageView 控件 沿 着 指定 的 折线 路 径 平 移 。 当 实例 运行 
之 后 , 蜂 蛛 图 像 (ImageView 控件 ) 位 于 屏幕 顶端 ,如 图 162. 1 的 左 图 所 示 ( 即 myscenel)。 单 击 屏 幕 ， 
则 蜂 蛛 图 像 将 沿 着 细 实 线 ( 实 际 不 可 见 ) 所 示 的 折线 路 径 平 移 到 屏幕 底部 ,如 图 162. 1 的 右 图 所 示 ( 即 
myscene2) 。 在 myscene2 中 单 击 屏 人 莫 , 则 蜂 蛛 图 像 将 沿 着 细 实 线 所 示 的 折线 路 径 ( 与 前 次 的 平移 方 癌 
相反 ) 平 移 到 屏幕 顶部 ,如 图 162. 1 的 左 图 所 示 ( 即 myscenel ) 。 
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// 单 击 myscenel 过 渡 到 myscene2( 图 像 沿 折线 向 下 平移 ) 
public void onClickScenel (View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myRootView); 
Scene myScene2 = Scene.getSceneForLayout(myRootView, R. layout.myscene2, this); 
TransitionManager. go(myScene2, TransitionInflater.from(MainActivity. this). 
inflateTransition(R. transition. mytransition) ); 
} 
// 单 击 myscene2 过 渡 到 myscenel( 图 像 沿 折线 向 上 平移 ) 
public void onClickScene2(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myRootView); 
Scene myScenel = Scene.getSceneForLayout(myRootView, R. layout.myscenel, this); 
TransitionManager. go(myScenel, TransitionInflater.from(MainActivity.this). 
inflateTransition(R. transition. mytransition) ); 
} 


上 面 这 段 代 码 在 MyCode\ MySample586 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,TransitionManager. go (myScene2，TransitionInflater. 
from(MainActivity. this). inflateTransition (R. transition。 mytransition)) 表示 使 用 在 mytransition 
过 渡 动 画 中 定制 的 ChangeBounds 动画 在 5 秒 内 沿 着 折线 路 径 从 myScenel 过 渡 到 myScene2 。 
mytransition 过 渡 动 画 的 主要 内 容 如 下 : 


<?xml version = "1.0”encoding = "utf 一 8"?> 

< transitionSet xmlns:android = "http://schemas. android. com/apk/res/android" 
android: duration = "5000" 
android: interpolator = " (Wandroid: interpolator/decelerate cubic"> 

< changeBounds > 

< patternPathMotion 
android: patternPathData = "M200 0 L50 250 L300 250 L200 500" /> 
</changeBounds > 
</transitionSet > 


上 面 这 段 代 码 在 MyCode\MySample586\app\src\main\res\transition\mytransition. XML 文件 
中 。 在 这 段 代 码 中 ,android:patternPathData 二 "M200 0 L50 250 L300 250 L200 500" 代 表 图 162. 1 
所 示 的 折线 数据 ,该 XML 文件 实际 上 相当 于 下 和 面 这 段 代码 : 


Path myPath = new Path( ); 

myPath. moveTo( 200, 0); 

myPath. lineTo( 50, 250); 

myPath. lineTo( 300, 250); 

myPath. lineTo( 200, 500); 

PatternPathMotion myMotion = new PatternPathMotion(); 
myMotion. setPatternPath(myPath); 

ChangeBounds myChangeBounds = new ChangeBounds(); 
myChangeBounds. setPathMotion(myMotion); 
myChangeBounds. setDuration( 5000); 


需要 说 明 的 是 ,transition 过 渡 动 画 通 第 需要 一 个 容 需 (也 叫 RootView) ,用 于 在 不 同 的 情形 下 加 
载 不 同 的 布局 。 在 MainActivity. java 文件 中 , myRootView 二 (ViewGroup) findViewById(R. id. 
myRootView) 即 是 用 于 根据 activity_main 布局 中 的 RelativeLayout 创建 动画 容器 。activity_main 布 
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局 的 主要 内 容 如 下 : 


<?xml] version = "1.0”encoding = "utf 一 8"?> 
< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns:tools = "http://schemas. android. com/tools" 
android: id= "(@ + id/activity_main" 
android:layout width= "match parent" 
android:layout height = "match parent" 
tools:context = "com. bin. luo. mysample. MainActivity"> 
<! 一 一 这 个 RelativeLayout 用 来 做 动画 的 父 布 局 -一 > 
< RelativeLayout android: id = "(@ + id/myRootView" 
android:layout width = "match parent" 
android:layout height = "wrap_content" 
android:layout centerInParent = "true"> 
< include layout = " (layout/myscenel"></ include > 
</RelativeLayout ></RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample586\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,< include layout 二" @ layout/myscenel" ></include > 表示 容 央 中 的 子 布 局 是 
myscenel ,myscenel 布局 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
android:onClick = "onClickScenel" 
android:padding = "20dp" 
android:gravity = "center horizontal" 
android: layout width= "match parent" 
android:layout height = "match parent"> 
< ImageView android: layout alignParentTop = "true" 
android: layout width = "100dp" 
android:layout height = "100dp" 
android:src = "(mipmap/myimagel" 
android: id= "(O + id/myImageViewl1" /> 
</RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample586\app\src\main\res\layout\myscenel. xml 文件 中 。 在 
使 用 TransitionManager. go() 方 法 实现 过 渡 动 画 时 ,通常 需要 两 个 布局 (场景 ) , 即 此 实例 的 myscenel 
和 myscene2 , 当 myscenel 的 某 个 控件 的 ID 值 与 myscene2 的 某 个 控件 的 ID 值 相同 , 且 位 置 或 大 小 
不 同时 ,在 从 myscenel 过 渡 到 myscene2 时 ,这 对 控件 就 会 执行 平移 或 缩放 动画 。myscene2 布局 的 
详细 内 容 如 下 : 


< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
android:padding = "20dp" 
android:gravity = "center horizontal" 
android:onClick = "onClickScene2" 
android: layout width= "match parent" 
android:layout height = "match parent"> 
< ImageView android: layout_width = "100dp" 
android: layout height = "100dp" 
android:src = "(@mipmap/myimagel" 
android: layout alignParentBottom= "true" 
android: id= "(@ + id/myImageView1l" /> 
</RelativeLayout > 
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上 面 这 段 代码 在 MyCode\MySample586\app\src\main\res\layout\myscene2. xml 文件 中 。 需 
要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 
25.0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample586 文件 夹 中 。 


163 ”使 用 TransitionManager 同时 实现 多 种 动画 


此 实例 主要 通过 在 XML 文件 中 将 explode 动画 和 fade 动画 添加 到 transitionSet 过 渡 动 画集 合 ， 
并 将 其 传递 到 TransitionManager 的 beginDelayedTransition() 方 法 中 ,实现 控件 在 执行 动画 时 产生 
explode 动画 和 fade 动画 的 登 加 效果 。 当 实例 运行 之 后 ,怪兽 图 像 (ImageVievw 控件 ) 位 于 屏幕 正中 ， 
单 击 屏幕 , 则 怪兽 图 像 将 以 随机 路 径 滑 出 屏幕 ,在 滑 出 屏幕 的 过 程 中 ,其 透明 度 将 渐渐 减弱 ,直到 完全 
消失 ; 再 次 单 击 屏幕 , 则 怪兽 图 像 将 以 随机 路 径 滑 和 人 屏幕 ,在 滑 人 屏幕 的 过 程 中 ,其 透明 度 将 渐渐 增 
强 , 直 到 完全 显示 ,效果 分 别 如 图 163. 1 的 左 图 和 右 图 所 示 。 
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图 163.1 
主要 代码 如 下 : 


// 单 击 屏 幕 , 图 像 将 组 合 explode 和 fade 两 种 动画 实现 显示 或 消失 
public void onClickScreen(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
TransitionManager. beginDelayedTransition (myRootView, TransitionInflater. from (MainActivity. this ) . 
inflateTransition(R. transition. mytransition) ); 
toggleVisibility(myImageView); 
} 
private static void toggleVisibility(View... views)!{ 
for (View v:views) { 
boolean isVisible = v.getVisibility() == View. VISIBLE; 
V. setVisibility(isVisible?View. INVISIBLE:View. VISIBLE); 
1 


上 F 面 这 段 代 码 在 MyCode\ MySample607\app\src\ main\java\com\bin\luo\mysample\ 
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MainActivity. java 文件 中 。 在 这 上段 代码 中， TransitionManager. beginDelayedTransition 
(myRootView, TransitionInflater. {from ( MainActivity. this ). inflateTransition ( R. transition. 
mytransition) ) 表 示 在 父 容器 myRootView 中 执行 mytransition 中 的 explode 和 fade 二 加 动画 。 
mytransition 过 渡 动 画 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 
<transitionSet xmlns:android = "http://schemas. android. com/apk/res/android" 
android: duration = "5000" 
android: interpolator = "(Qandroid: interpolator/decelerate cubic"> 
< explode/>< fade/> 
</transitionSet > 


上 和 面 这 段 代 码 在 MyCode\MySample607\app\src\main\res\transition\mytransition. xml 文件 
中 。 在 这 段 代 码 中 ,< explode/> 即 表示 explode 动画 ,可 以 在 标签 内 设置 其 属性 ; 同 理 , 也 可 以 在 < 
fade/> 标 签 内 设置 fade 动画 的 属性 。 在 Android 中 , Fade、 Explode、Slide 动画 作用 于 View 的 
Visibility 属性 改变 的 时 候 。transition 过 渡 动 画 通 常 需 要 一 个 容 磊 (也 岂 RootView) ,用 于 在 不 同 的 
情形 下 加 载 不 同 的 布局 。 在 MainActivity. java 文件 中 ,myRootView 二 (ViewGroup)findViewByld 
(R. id. myRootView) 用 于 根据 activity_ main 布局 中 的 RelativeLayout 创建 动画 容 右 。activity_main 
布局 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
< RelativeLayout xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns:tools = "http://schemas. android. com/tools" 
android: id = "(@ + id/activity main" 
android:layout width= "match parent" 
android:layout height = "match parent" 
android:onClick = "onClickScreen" 
tools:context = "com. bin. luo. mysample. MainActivity"> 
<! 一 一 这 个 RelativeLayout 用 来 做 动画 的 父 布局 -一 > 
< RelativeLayout android: id = "(@ + id/myRootView" 
android:gravity = "center horizontal |center vertical" 
android:layout width= "match parent" 
android:layout height = "match parent"> 
< ImageView android: layout width = "150dp" 
android:layout height = "200dp" 
android: scaleType = "centerCrop" 
android: src = "(mipmap/myimagel" 
android: id = "(@ + id/myImageView" /> 
</RelativeLayout ></RelativeLayout > 


上 面 这 段 代 码 在 MyCode\MySample607\app\src\main\res\layout\activity_main. xml 文件 中 。 
需要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 
25.0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample607 文件 夹 中 。 


164 ”使 用 TransitionManager 实现 XML 定制 动画 


此 实例 主要 通过 使 用 过 渡 动 画 (fade) 的 targetId 属性 指定 控件 ,实现 在 两 个 布局 (myscenel 和 
myscene2) 切 换 时 , 仅 有 指定 的 控件 执行 过 渡 动 画 。 实 例 运 行 之 后 , 当 从 myscenel 切换 到 myscene2 
时 , 仅 有 myImageView3 控件 ( 即 下 面 的 电影 海报 图 像 ) 执 行 淡 入 动画 ,如 图 164. 1 的 左 图 所 示 ( 即 
myscene2)。 当 从 myscene2 切换 到 myscenel 时 ,也 仅 有 myImageView3 控件 ( 即 下 面 的 电影 海报 图 
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像 ) 执 行 淡出 动画 ,如 图 164. 1 的 右 图 所 示 ( 即 myscenel ) 。 
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// 单 击 myscenel 过 渡 到 myscene2( 仅 下 面 的 图 像 (myImageView3 ) 执 行 过 渡 动 画 ) 
public void onClickScenel (View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
Scene myScene2 = Scene.getSceneForLayout(myRootView, R. layout.myscene2, this); 
TransitionManager. go(myScene2, TransitionInflater.from(MainActivity.this). 
inflateTransition(R.transition. mytransition) ); 
} 
// 单 击 myscene2 过 渡 到 myscenel( 仅 下 面 的 图 像 (myImageView3) 执 行 过 渡 动 画 ) 
public void onClickScene2(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id. myRootView); 
Scene myScenel = Scene.getSceneForLayout(myRootView, R. layout.myscenel, this); 
TransitionManager. go (myScenel, TransitionInflater. from (MainActivity. this). inflateTransition(R. 
transition. mytransition) ); 
} 


上 面 这 上段 代码 在 MyCode\ MySample590\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,TransitionManager. go (myScene2，TransitionInflater. 
from(MainActivity. this). inflateTransition( R. transition. mytransition) ) 表 示 在 从 myScenel 切换 到 
myScene2 时 ,执行 在 mytransition 过 渡 动 画 中 定制 的 myImageView3 控件 淡 入 动画 。mytransition 
过 渡 动 画 的 主要 内 容 如 下 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 
<transitionSet xmlns:android = "http://schemas. android. com/apk/res/android" 
android: duration = "5000" 
android: interpolator = "(Qandroid: interpolator/decelerate cubic"> 
<! -- 仅 myImageView3 执行 淡 入 过 渡 动 画 -- > 
< fade android: fadingMode = "fade in"> 
<targets>< target android:targetId = " (Qid/myImageView3" /></targets > 
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</fade > 
<! -- 仅 myImageView3 执行 淡出 过 渡 动 画 -- > 
< fade android: fadingMode = "fade out"> 
<targets >< target android:targetId = " (Qid/myImageView3" /></targets > 
</fade > 
</transitionSet > 


上 面 这 段 代 码 在 MyCode\MySample590\app\src\main\res\transition\mytransition. xml 文件 
中 。 在 这 段 代 码 中 ,< target android:targetId 王 "id/myImageView3" /> 表示 在 目标 布局 myScene2 
中 , 仅 有 myImageView3 控件 执行 痰 入 (或 淡出 ?动画 ,其 他 控件 则 按照 默认 的 方式 进行 切换 。 过 渡 动 
男 通 和 常 需 要 一 个 容器 (也 叫 RootView) ,用 于 在 不 同 的 情形 下 加 载 不 同 的 布局 。 关 于 activity_main、 
myscenel .myscene2 三 个 布局 的 详细 内 容 请 在 源 代码 中 查看 MyCode\MySample590\app\src\main\ 
res\layout\activity_main. xml 文件 .MyCode\ MySample590\app\src\main\res\layout\myscenel. 
xml 文件 和 MyCode\MySample590\app\ src\main\res\layout\myscene2. xml 文件 。 需 要 说 明 的 是 ， 
使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 25. 0. 1' 依 赖 项 。 
此 实例 的 完整 项 目 在 MyCode\MySample590 文件 夹 中 。 


165 ”使 用 TransitionManager 指定 控件 执行 动画 


此 实例 主要 通过 使 用 过 渡 动 画 类 的 addTarget() 方 法 指定 需要 执行 过 渡 动 画 的 控件 ,实现 两 个 布 
局 在 使 用 过 渡 动 画 切 换 时 , 仅 有 指定 控件 执行 过 渡 动 画 , 其 他 控件 则 不 执行 过 渡 动 画 。 当 实例 运行 之 
后 , 单 击 activity_main 布局 的 ImageView 控件 (电影 海报 图 像 ) ,如 图 165. 1 的 左 图 所 示 , 则 将 以 Fade 
动画 风格 切换 到 activity_second 布局 ; 在 此 实例 中 ,由 于 activity_second 布局 上 面 的 ImageView 控 
件 《 风 语 者 电影 海报 图 像 ) 被 指定 执行 过 渡 动 画 , 因 此 在 从 activity_main 布局 切换 到 activity _ 
second 布局 时 , 仅 activity_second 布局 上 面 的 ImageView 控件 (《 风 语 者 ) 电 影 海 报 图 像 ) 有 Fade 过 渡 
动画 效果 ,activity_second 布局 下 面 的 ImageView 控件 (电影 海报 图 像 ) 则 没有 Fade 过 渡 动 画 效 果 ， 
如 图 165. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickmyImageEnter(View v) { // 响 应 单 击 图 像 进 入 activity_second 布局 
ViewGroup myMainLayout = (ViewGroup)findViewById(R. id.activity main); 
Scene mySecondScene = Scene.getSceneForLayout(myMainLayout, 
R. layout.activity second, this); 
Fade myFade = new Fade(); 


myFade. setDuration( 5000); 
myFade. addTarget(R. id.myImageView2 ) ; // 仅 在 myImageView2 控件 上 执行 过 渡 动 画 
TransitionManager. go(mySecondScene, myFade); 

} 

public void onClickmyImageExit(View v) { // 响 应 单 击 图 像 进入 activity_main 布局 


ViewGroup mySecondLayout = (ViewGroup)findViewById(R. id.activity second); 
Scene myMainScene = Scene. getSceneForLayout(mySecondLayout, 
R. layout. activity main, this); 
Fade myFade = new Fade( ); 
myFade. setDuration( 5000); 
TransitionManager. go(myMainScene, myFade); 


上 面 这 上段 代码 在 MyCode\ MySample579\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myFade. addTarget (R. id. myImageView2 ) 表示 
myImageView2 控件 必须 执行 myFade 过 渡 动 画 。TransitionManager. go(mySecondScene, myFade) 
表示 在 从 activity_main 布局 切换 到 activity_second 布局 时 ,执行 myFade 规定 的 过 渡 动 画 。 关 于 
activity_main ,activity_second 布局 的 详细 内 容 请 参考 源 代 码 中 的 MyCode\MySample579\app\src\ 
main\res\layout\activity_main. xml 文件 和 MyCode\ MySample579\app\src\main\res\layout\ 
lavyout second. xml 文件 。 此 实例 的 完整 项 目 在 MyCode\MySample579 文件 夹 中 。 


166 ”使 用 TransitionManasger 实现 列表 项 请 入 动画 


此 实例 主要 通过 在 TransitionManager 的 beginDelayedTransition () 方 法 中 设置 RecyclerView 
控件 为 过 渡 动 画 Slide 的 容 髓 ,实现 在 显示 RecyclerView 时 出 现 各 个 Item( 列 表 项 ) 从 右 端 滑 入 的 动 
男 效 果 , 以 及 在 隐藏 RecyclerView 时 出 现 各 个 Item( 列 表 项 ) 回 右 端 滑 出 的 动画 效果 。 当 实例 运行 之 
后 , 单 击 “执行 Slide 动画 ?按钮 , 则 RecyclerView 的 各 个 Item 将 回 右 端 滑 出 ,直到 完全 消失 ; 再 次 单 
击 “ 执 行 Slide 动画 ”按钮 , 则 RecyclerView 的 各 个 Item 将 从 右 端 滑 入 ,直到 完全 显示 ,效果 分 别 如 
图 166. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickmyBtnl1 (View v) { // 响 应 单 击 "执行 Slide 动画 "按钮 
TransitionSet myTransitionSets = new TransitionSet(); 
Slide mySlide = new Slide(); 
mySlide. setSlideEdge( Gravity. RIGHT ) ; 
myTransitionSets. addTransition(mySlide); 
myTransitionSets. setDuration(2000); 
TransitionManager. beginDelayedTransition(myRecyclerView, myTransitionSets); 
if (bChanged) { myRecyclerView. setAdapter(myAdapter); } 
else { myRecyclerView. setAdapter(null); } 
bChanged = !bChanged; 
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F 面 这 段 代 码 在 MyCode\ MySample609\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 ,。 在 这 上 段 代码 中， TransitionManager. beginDelayedTransition 
(myRecyclerView, myTransitionSets) 表 示 在 myRecyclerView 中 执行 myTransitionSets 中 的 Slide 
动画 。Slide 动画 通常 在 显示 或 隐藏 控件 的 时 候 就 执行 ,无 须 其 他 代码 。 在 执行 Slide 动画 时 通常 需 
要 一 个 容器 (一 般 为 布局 管理 器 ) ,但 是 RecyclerView 控件 天 然 就 是 一 个 容器 , 它 里 面 可 以 容纳 多 个 
Item, 因 此 也 无 须 其 他 代码 ,从 activity_main 布局 中 就 可 以 看 出 实现 此 动画 没有 其 他 的 容器 。 关 于 
activity_main 布局 的 详细 内 容 请 参考 源 代码 中 的 MyCode\MySample609\app\src\main\res\layout\ 
activity_main. xml 文件 。 需 要 说 明 的 是 ,在 Android Studio 中 ,使 用 此 实例 的 相关 类 和 控件 需要 在 
gradle 中 引入 compile 'com. android. support:recyclerview-v7:25. 2.0' 依 赖 项 。 此 实例 的 完整 项 目 在 
MyCode\MySample609 文件 夹 中 。 


MySample MySample 


执行 Slide 动画 执行 Slide 动画 


日 常生 活 中 荒诞 行为 的 心理 学 解读 日 常生 活 中 荒诞 行为 的 心理 学 解读 
以 最 有 效 的 数字 呈现 技巧 征服 你 的 观众 | 以 最 有 效 的 数字 呈现 技巧 征服 你 的 观众 
不 确定 性 情景 下 的 心理 预期 与 行为 选择 ‖ 不 确定 性 情景 下 的 心理 预期 与 行为 选择 
算术 与 几何 的 妙趣 算术 与 几何 的 妙趣 
不 可 思议 的 科技 史 不 可 思议 的 科技 史 
一 首 献 给 数学 的 情歌 一 首 献 给 数学 的 情歌 
学 会 像 数 学 家 一 样 思考 学 会 像 数 学 家 一 样 思考 
奇妙 数学 的 100 个 重大 突破 奇妙 数学 的 100 个 重大 突破 
解决 问题 的 非常 规 思考 法 解决 问题 的 非常 规 思 考 法 


生活 中 的 概率 趣事 生活 中 的 概率 趣事 


图 166.1 


167 ”使 用 TransitionManager 实现 引线 路径 动 画 


此 实例 主要 通过 在 TransitionManager 的 go() 方 法 中 加 载 在 XML 文件 中 定制 的 弧 线路 径 动 画 ， 
实现 在 两 个 布局 (myscenel 和 myscene2) 切 换 时 ,ImageView 控件 沿 着 指定 的 弧 线路 径 平 移 。 当 实例 
运行 之 后 ,蜘蛛 图 像 (ImageView 控件 ) 位 于 屏幕 左 端 , 如 图 167. 1 的 左 图 所 示 ( 即 myscenel)。 单 击 屏 
幕 , 则 蜂 蛛 图 像 将 沿 着 细 实 线 所 示 的 弧 线 路 径 平 移 到 屏幕 右 端 ,如 图 167. 1 的 右 图 所 示 ( 即 
myscene2)。 在 myscene2 中 单 击 屏 莫 , 则 蜂 蛛 图 像 将 沿 着 细 实 线 所 示 的 弧 线 路 径 平 移 到 屏幕 左 端 ,如 
图 167. 1 的 左 图 所 示 ( 即 myscenel ) 。 

主要 代码 如 下 : 


// 单 击 myscenel 过 渡 到 myscene2( 图 像 沿 着 上 弧 线 向 右 平移 ) 
public void onClickScenel (View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
Scene myScene2 = Scene.getSceneForLayout(myRootView, R. layout.myscene2, this); 
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TransitionManager. go(myScene2, TransitionInflater.from(MainActivity.this). 
inflateTransition(R. transition. mytransition) ); 
} 
// 单 击 myscene2 过 渡 到 myscenel( 图 像 沿 着 下 弧 线 向 左 平移 ) 
public void onClickScene2(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
Scene myScenel = Scene.getSceneForLayout(myRootView, R. layout.myscenel, this); 
TransitionManager.go(myScenel, TransitionInflater.from(MainActivity. this). 
inflateTransition(R.transition. mytransition) ); 
} 


上 和 面 这 段 代码 在 MyCode\ MySample588\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,TransitionManager. go (myScene2，TransitionInflater. 
from(MainActivity. this). inflateTransition (R. transition. mytransition) ) 表示 使 用 在 mytransition 
过 渡 动 画 中 定制 的 ChangeBounds 动画 在 5 秒 内 沿 着 弧 线 路 径 从 myScenel 过 渡 到 myScene2 。 
mytransition 过 渡 动 画 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "utf 一 8"?> 
<transitionSet xmlns:android = "http://schemas. android. com/apk/res/android" 
android: duration = "5000" 
android: interpolator = "(Qandroid: interpolator/decelerate cubic"> 
< ChangeBounds > 
< patternPathMotion android: patternPathData= "M8,10 a4,601,166"/> 
</changeBounds ></transitionSet > 


上 和 面 这 段 代 码 在 MyCode\MySample588\app\src\main\res\transition\mytransition. XML 文件 
中 。 在 这 段 代码 中 ,android:patternPathData 二 "M8,10 a4,6 0 1,1 6 6" 代 表 图 167. 1 所 示 的 弧 线 数 
据 。< changeBounds > 标签 表示 使 用 ChangeBounds 动画 。transition 过 渡 动 画 通 常 需 要 一 个 容 胡 (也 
叫 RootView) ,用 于 在 不 同 的 情形 下 加 载 不 同 的 布局 。 关 于 此 实例 的 activity_main、myscenel、 
myscene2 三 个 布局 的 详细 内 容 请 参考 源 代码 中 的 MyCode\ MySample588\app\src\main\res\layout 


\activity_main. xml 文件 .MyCode\ MySample588\ app\src\main\res\layout\myscenel. xml 文件 和 
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MyCode\MySample588\ app\src\main\res\ layout\myscene2. xml 文件 。 需 要 说 明 的 是 ,使 用 此 实 
例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design:25. 0.1' 依 赖 项 。 此 实例 的 
完整 项 目 在 MyCode\MySample588 文件 夹 中 ， 


168 ”使 用 TransitionManager 实现 裁剪 区 域 动画 


此 实例 主要 通过 在 TransitionManager 的 go() 方 法 中 设置 ChangeClipBounds 动画 ,实现 在 两 个 
布局 (myscenel 和 myscene2) 过 渡 时 ,裁剪 的 矩形 区 域 将 实现 平移 动画 。 当 实例 运行 之 后 , 单 击 屏幕 ， 
则 在 绿色 背景 左上 角 的 裁剪 矩形 将 平移 到 右 下 角 , 如 图 168. 1 的 右 图 所 示 ( 即 myscene2)。 在 
myscene2 中 单 击 屏幕 , 则 在 绿色 背景 右 下 角 的 裁剪 矩形 将 平移 到 左上 和 角 , 如 图 168. 1 的 左 图 所 示 ( 即 


myscenel ) 。 


< 
tes 


and the Terrible, Horrible, and the Terrible, Horrible, 
NO GOOD, VERY BAD DAY NO GOOD, VERY BAD DAY 


图 168.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

ImageView mylImageViewl, mylImageView2,; 

ViewGroup myRootView; 

Scene myScenel, myScene2; 

View myInflatel,myInflate2,; 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myRootView = (ViewGroup)findViewById(R. id.myRootView); 
myInflatel = LayoutInflater. from(this). inflate(R. layout.myscenel, null); 
myScenel = new Scene(myRootView,myInflatel ) ; 
myImageViewl = (ImageView) myInflatel.findViewById(R. id. myImageViewl ) ; 
myImageViewl. setClipBounds(new Rect(0,0,300,400)); 
TransitionManager. go(myScenel ) ; 


} 

// 单 击 myscenel 过 渡 到 myscene2( 裁 剪 矩形 从 左上 和 角 平 移 到 右 下 角 ) 

public void onClickScenel(View v) { 
myInflate2 = LayoutInflater.from(this). inflate(R. layout.myscene2, null); 
myScene2 = new Scene(myRootView,myInflate2); 
myImageView2 = (ImageView) myInflate2.findViewById(R. id.myImageView2); 
myImageView2. setClipBounds(new Rect(300,400,600,800)); 
TransitionManager. go(myScene2, new ChangeClipBounds(). setDuration(5000)); 

} 

// 单 击 myscene2 过 渡 到 myscenel (裁剪 矩形 从 右 下 角 平 移 到 左上 角 ) 

public void onClickScene2(View v) { 


TransitionManager. go(myScenel, new ChangeClipBounds(). setDuration(5000)); 
rs 


上 面 这 段 代 码 在 MyCode\ MySample583\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 ，TransitionManager. go (myScene2， new 
ChangeClipBounds(). setDuration(5000)) 表 示 以 ChangeClipBounds 动画 风格 ( 即 裁剪 矩形 平移 ) 在 5 
秒 内 从 myScenel 过 渡 到 myScene2 。ChangeClipBounds 与 ChangeBounds 的 过 渡 动 画 功 能 比较 类 
似 , 不 同 的 是 ,ChangeBounds 针对 的 是 view 控件 ,而 ChangeClipBounds 针对 的 是 view 控件 的 剪 切 
区 域 ,即使 用 setClipBound(Rect rect) 方 法 设置 的 rect, 如 果 没 有 设置 rect, 则 没有 动画 效果 。 关 于 此 
实例 的 activity_main、myscenel、myscene2 三 个 布局 的 详细 内 容 请 参考 源 代码 中 的 MyCodeA\ 
MySample583\app\src\main\res\layout\ activity_main. xml 文件 .MyCode\MySample583\app\src\ 
main\ res \layout \ myscenel. xml 文件 和 MyCode\ MySample583\app\src\ main\res\layout\ 
myscene2. xml 文件 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. 
android. support:design: 25. 0.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample583 文件 夹 中 。 


169 ”通过 设置 和 获取 控件 的 Tag 人 确定 动画 过 渡 行 为 


此 实例 主要 通过 使 用 setTag() 和 getTag() 方 法 为 控件 添加 标记 和 获取 控件 的 标记 ,使 控件 实现 
根据 当前 的 标记 决定 如 何 执行 过 渡 动 画 。 当 实例 运行 之 后 , 单 击 ImageView 控件 (电影 海报 图 像 ) ,由 
于 ImageView 控件 的 当前 标记 是 myimagel ,因此 执行 从 myimagel 透明 过 渡 到 myimage2 的 动画 ; 
再 次 单 击 ImageView 控件 ,由 于 此 时 ImageView 控件 的 当前 标记 已 经 被 设置 为 myimage2 ,因此 执行 
从 myimage2 透明 过 渡 到 myimagel 的 动画 ,效果 分 别 如 图 169. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickImage(View v) { // 单 击 图 像 之 后 透明 过 渡 到 另 一 幅 图 像 ,或 者 相反 
// 如 果 当 前 ImageView 控件 加 载 图 像 是 myimagel, 则 从 myimagel 过 渡 到 myimage2 
if ((Integer) myImageView. getTag() == R.mipmap.myimagel) { 
myTransitionDrawable = new TransitionDrawable(new Drawable[ ] 
{getDrawable(R.mipmap. myimagel), getDrawable(R.mipmap. myimage2)}); 
myImageView. setTag(R. mipmap. myimage2 ); // 设 置 标记 myimage2 
// 如 果 当 前 ImageView 控件 加 载 图 像 是 myimage2, 则 从 myimage2 过 渡 到 myimagel 
} else if ((Integer) myImageView. getTag() == R.mipmap.myimage2) { 
myTransitionDrawable = new TransitionDrawable(new Drawable[ ] 
{getDrawable(R. mipmap. myimage2), getDrawable(R.mipmap. myimagel )}); 
myImageView. setTag(R. mipmap.myimagel ); // 设 置 标记 myimagel 
} 
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myImageView. setImageDrawable(myTransitionDrawable); 
myTransitionDrawable. startTransition(2000);  // 设 置 动 画 时 间 为 2 秒 
} 


中 加 划 :二 "i 日 93% 下 午 5:37 


169. 1 


上 面 这 段 代 码 在 MyCode\ MySample642\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myImageView. setTag(R. mipmap. myimagel ) 用 于 在 
ImageView 控件 上 设置 标记 ,myImageView. getTag() 用 于 获取 在 ImageView 控件 上 设置 的 标记 ,一 
般 情 况 下 ,getTag() 方 法 的 返回 值 应 该 与 setTag() 方 法 的 参数 值 的 数据 类 型 一 致 。 此 实例 的 完整 项 
日 在 MyCode\MySample642 文件 夹 中 。 


170 ”在 TransitionSet 中 指定 多 个 动画 的 执行 顺序 


此 实例 主要 通过 设置 TransitionSet 的 setOrdering() 方 法 的 参数 ,实现 在 两 个 布局 过 渡 的 过 程 中 
多 个 动画 或 是 同时 执行 ,或 是 顺序 执行 。 当 实例 运行 之 后 ,Revolting Rhymes 电影 海报 图 像 位 于 屏幕 
的 正中 ,如 图 170. 1 的 左 图 所 示 ( 即 myscenel); 单 击 屏幕 , 则 将 淡出 Revolting Rhymes 电影 海报 图 
像 ,然后 淡 入 Trolls Holiday 电影 海报 图 像 ( 即 myscene2) ,如 图 170. 1 的 右 图 所 示 。 在 myscene2 中 
单 击 屏幕 , 则 将 淡出 Trolls Holiday 电影 海报 图 像 ,然后 淡 入 Revolting Rhymes 电影 海报 图 像 , 如 
图 170. 1 的 左 图 所 示 。 

主要 代码 如 下 : 


// 单 击 myscenel 过 渡 到 myscene2( 先 淡出 myscenel, 后 淡 入 myscene2) 
public void onClickScenel (View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
Scene myScene2 = Scene.getSceneForLayout(myRootView, R. layout.myscene2, this); 
Fade myFadeIn = new Fade( Fade. IN); 
Fade myFadeOut = new Fade(Fade. OUT); 
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TransitionSet myTransitionSet = new TransitionSet( ); 
myTransitionSet.addTransition(myFade0ut). addTransition(myFadeIn); 
myTransitionSet. setOrdering(TransitionSet. ORDERING_SEOUENTIRL ) ; 
myTransitionSet. setDuration( 5000); 
TransitionManager.go(myScene2, myTransitionSet); 
} 
// 单 击 myscene2 过 渡 到 myscenel( 先 淡出 myscene2, 后 淡 入 myscenel ) 
public void onClickScene2(View v) { 
ViewGroup myRootView = (ViewGroup)findViewById(R. id.myRootView); 
Scene myScenel = Scene.getSceneForLayout(myRootView, R. layout.myscenel, this); 
Fade myFadelIn = new Fade(Fade. IN); 
Fade myFadeOut = new Fade(Fade. OUT); 
TransitionSet myTransitionSet = new TransitionSet( ); 
myTransitionSet.addTransition(myFadeOut).addTransition(myFadelIn); 
myTransitionSet. setOrdering(TransitionSet. ORDERING_SEOUENTIRL ) ; 
myTransitionSet. setDuration(5000); 
TransitionManager.go(myScenel, myTransitionSet); 


上 面 这 段 代 码 在 MyCode\MySample555\app\src\main\java\com\bin\luo\mysample\ MainActivity. 

java 文件 中 。 在 这 段 代 码 中 ,myTransitionSet. addTransition (myFadeOut). addTransition (myFadeIn) 用 
于 在 myTransitionSet 过 渡 动 画集 合 中 增加 myFadeOut 和 myFadeIn 动画 。myTransitionSet. setOrdering 
(TransitionSet. ORDERING_SEQUENTIAL) 表 示 此 实例 的 两 个 淡出 淡 和 动画 将 按照 添加 顺序 执行 ,如 
果 是 myTransition . setOrdering (TransitionSet. ORDERING_TOGETHER) , 则 两 个 动画 将 同时 执行 。 关 
于 activity_main、myscenel ,myscene2 三 个 布局 的 详细 内 容 请 参考 源 代 码 中 的 MyCode\MySample555\app 
\src\ main\res\layout\activity_main. xml 文件 MyCode\ MySample555\app\src\main\res\ layout\ 
myscenel. xml 文件 和 MyCode\MySample555\app\src\main\res\layout\myscene2. xml 文件 。 此 外 ,使 用 
此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android，support: design:25. 0.1' 依 赖 项 。 此 实例 的 
完整 项 目 在 MyCode\MySample555 文件 夹 中 。 
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171 使 用 TransitionDrawable 透明 切换 两 由 图 像 


此 实例 主要 通过 使 用 TransitionDrawable 创建 图 像 过 渡 动 画 , 实 现在 指定 的 时 间 内 从 一 幅 图 像 
切换 到 另 一 幅 图 像 。 当 实例 运行 之 后 , 单 击 屏幕 , 则 在 5 秒 内 从 一 幅 电 影 海 报 图 像 渐 渐 切 换 到 (改变 
透明 度 ) 另 一 幅 电 影 海 报 图 像 , 效 果 分 别 如 图 171. 1 的 左 图 和 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickImage(View v) { // 单 击 图 像 之 后 透明 过 渡 到 另 一 幅 图 像 
TransitionDrawable myTransitionDrawable = 
(TransitionDrawable)getResources( ) . getDrawable(R. drawable. mytransition); 
ImageView myImageView = (ImageView)findViewById(R. id. myImageView); 
myImageView. setImageDrawable( myTransitionDrawable); 
myTransitionDrawable. startTransition(5000); 
} 


上 面 这 段 代 码 在 MyCode\ MySample541\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 , myTransitionDrawable 二 (TransitionDrawable) 
getResources(). getDrawable( R. drawable. mytransition) 中 的 mytransition 用 于 设置 淡 和 淡出 过 渡 
动画 的 源 图 像 和 目标 图 像 。mytransition 过 渡 动画 的 主要 内 容 如 下 : 


< transition xmlns:android = "http://schemas. android. com/apk/res/android"> 
< item android: drawable = " (Qdrawable/myimagel" /> 
< item android: drawable = " (Odrawable/myimage2" /> 

</transition> 


上 和 面 这 上段 代码 在 MyCode\MySample541\app\src\main\res\drawable\mytransition. xml 文件 
中 。< item android:drawable 二"(@ drawable/myimagel"/ > 表示 切换 的 源 图 像 是 myimagel,< item 
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android:drawable 二 "(@drawable/myimage2"/> 表 示 切 换 的 目标 图 像 是 myimage2。 此 实例 的 完整 项 
目 在 MyCode\MySample541 文件 夹 中 。 


172 ”使 用 AnimatedVectorDrawable 实现 转圈 动画 


此 实例 主要 通过 使 用 AnimatedVectorDrawable 加 载 在 SVG 中 实现 的 转圈 动画 ,并 作为 
ImageView 控件 的 背景 ,实现 高 仿 Chrome 进度 条 的 转圈 加 载 效 果 。 当 实例 运行 之 后 ,屏幕 中 心 的 天 
蓝 色 进度 条 将 沿 着 顺 时 针 方 品 进 行 旋转 ,并 伴 有 视差 ,类 似 于 Chrome 浏览 器 的 标签 页 加 载 进度 条 , 效 
果 分 别 如 图 172. 1 的 左 图 和 右 图 所 示 。 


So 


MySample MySample 


图 172.1 


public class MainActivity extends Activity { 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
ImageView myImageView = (ImageView) findViewById(R. id.myImageView) ; 
// 动 态 加 载 SVG 动画 
AnimatedVectorDrawable myAnimatedVectorDrawable = 
(AnimatedVectorDrawable) getDrawable(R. drawable. mysvganim); 
// 设 置 ImageView 控件 的 背景 为 SVG 动画 
myImageView. setBackground(myAnimatedVectorDrawable); 
myAnimatedVectorDrawable. start( ); // 启 动 动画 
} } 


上 和 面 这 段 代码 在 MyCode\MySample870\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 ,myAnimatedVectorDrawable= (AnimatedVectorDrawable) getDrawable(R. 
drawable. mysvganim) 用 于 根据 矢量 动画 mysvganim 创建 AnimatedVectorDrawable 对 象 , 以 实现 转圈 动 
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男 。 矢 量 动画 mysvganim 是 一 个 XML 格式 的 文件 , 它 的 主要 作用 是 将 动画 XML 文件 和 矢量 图 形 XML 
文件 组 合 在 一 起 ,矢量 动画 mysvganim 的 主要 内 容 如 下 : 


< animated - vector xmlns:android = "http://schemas.android. com/apk/res/android" 
android: drawable = " (Wdrawable/myprogressbar"> 
<! -- 该 动画 通过 更 改 SVG 图 形 的 路 径 对 象 的 trimPathOffset 值 , 以 实现 匀速 旋转 特效 -- > 
<target android: name = "myprogressbar" 
android: animation = "(Wanimator/myrotateanim" /> 
<! -- 该 动画 通过 更 改 SVG 图 形 的 路 径 对 象 的 trimPathEnd 值 , 以 实现 旋转 视差 特效 --> 
<target android: name = "myprogressbar" 
android: animation = " (Oanimator/myrotateaccelerateanim" /> 


</animated - vector > 


上 面 这 段 代码 在 MyCode\MySample870\app\src\main\res\drawable\mysvganim. xml 文件 中 。 
在 这 段 代 码 中 ,android:drawable 王 "drawable/myprogressbar" 的 myprogressbar 是 矢量 图 形 文 件 。 
myprogressbar. xml 文件 的 主要 内 容 如 下 : 


< Vector xmlns:android = "http://schemas. android. com/apk/res/android" 
android:width = "200dp" 
android: height = "200dp" 
android: viewportHeight = "200.0" 
android: viewportWidth = "200.0"> 
< path android: name = "myprogressbar" 
android:fillColor = " # fff" 
android: pathData = " M100, 100m - 40,0a40,40 0,1 1,80 0a40,40 0,1 1, - 80 0" 
android: strokeColor = " # 00BFFF" 
android: strokeWidth = "5" 
android:trimPathStart = "0.5"/> 
</vector > 


上 面 这 段 代 码 在 MyCode\MySample870\app\src\main\res\drawable\myprogressbar. xml 文件 
中 。 在 mysvganim. xml 文件 中 ,android:animation 一 "(@animator/myrotateanim "的 myrotateanim 是 
动画 文件 ,该 动画 通过 更 改 SVG 图 形 的 路 径 对 象 的 trimPathOffset 值 , 以 实现 勺 速 旋 转 特效 。 动 画 
文件 myrotateanim. xml 的 主要 内 容 如 下 : 


< objectRnimator xmlns:android = "http://schemas.android. com/apk/res/android" 
android: duration = "2500" 
android: interpolator = " (Wandroid: interpolator/ linear" 
android: propertyName = "trimPathOffset" 
android: repeatCount =" - 1" 
android: valueFrom = "0" 
android:valueTo = "1" 
android: valueType = "floatType" /> 


上 面 这 段 代码 在 MyCode\MySample870\app\src\main\res\animator\myrotateanim. xml 文件 
中 。 在 mysvganim. xml 文件 中 , android: animation 三 " (@ animator/ myrotateaccelerateanim" 的 
myrotateaccelerateanim 是 动画 文件 ,该 动画 文件 通过 更 改 SVG 图 形 的 路 径 对 象 的 trimPathEnd 值 ， 
以 实现 旋转 视差 特效 o myrotateaccelerateanim. xml 动画 文件 的 主要 内 容 如 下 : 


第 5 章 《Ce 


< objectAnimator xmlns:android = "http://schemas. android. com/apk/res/android" 
android: duration = "2500" 
android: interpolator = " (Wandroid: anim/accelerate decelerate interpolator" 
android: propertyName = "trimPathEnd" 
android: repeatCount =" -1" 
android: valueFrom= "0" 
android:valueTo= "1" 
android:valueType = "floatType" /> 


上 面 这 段 代 码 在 MyCode\MySample870\app\src\main\res\animator\ myrotateaccelerateanim. 
xml 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample870 文件 夹 中 。 


173 ”创建 AnimatedVectorDrawableCompat 动 团 


此 实例 主要 通过 设置 动画 集合 (Cset) 的 android:ordering 属性 为 together, 实 现 同时 执行 该 集合 中 
的 所 有 动画 。 当 实例 运行 之 后 , 单 击 “ 执 行动 画 ” 按 钮 , 则 矢量 图 形 “、/ ”符号 将 同时 执行 rotation、 
scaleX scaleY 三 种 动画 ,效果 分 别 如 图 173. 1 的 左 图 和 右 图 所 示 。 


MySample 


图 173.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

ImageView myImageView; 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myImageView = (ImageView)findViewById(R. id.myImageView); 
AnimatedVectorDrawableCompat myAnimatedVectorDrawableCompat = 

AnimatedVectorDrawableCompat. create(this, R. drawable. myanimated); 

mylImageView. setImageDrawable( myAnimatedVectorDrawableCompat ); 
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} 
public void onClickmyBtnl(View v) { // 响 应 单 击 "执行 动画 "按钮 
((Animatable) myImageView. getDrawable( ) ) . start( ); 

上 三 


上 面 这 段 代码 在 MyCode\MySample613\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 ,myAnimatedVectorDrawableCompat 三 AnimatedVectorDrawableCompat. 
create(this,R. drawable. myanimated) 用 于 加 载 动 画 文件 myanimated. xml 中 的 矢量 动画 。myanimated. 
xml 动画 文件 的 主要 内 容 如 下 : 


< animated — vector xmlns:android = "http://schemas. android. com/apk/res/android" 
xmlns:aapt = "http://schemas. android. com/aapt" 
android:drawable = "(@drawable/myvector"> 
< target android:name = "mytick"> 
<aapt:attr name = "android:animation"> 
< Set android:ordering = "together"> 
<objectAnimator android:duration = "500" 
android: interpolator = "(Qandroid: interpolator/linear" 
android: propertyName = " rotation" 
android:valueFrom = "180" 
android:valueTo = "0"/> 
<objectAnimator android:duration = "500" 
android: interpolator = "(Qandroid: interpolator/linear" 
android: propertyName = " scaleX" 
android:valueFrom = "0.2" 
android:valueTo = "1"/> 
< objectAnimator android:duration = "500" 
android: interpolator = "(@android: interpolator/linear" 
android: propertyName = " scaleY" 
android:valueFrom = "0.2" 
android:valueTo = "1"/> 
</ set ></aapt:attr ></target ></animated — vector > 


上 面 这 段 代码 在 MyCode\MySample613\app\src\main\res\drawable\myanimated. xml 文件 中 。 
在 这 段 代 码 中 ,android:ordering 二 "together" 表 示 动 画集 合 中 的 多 个 动画 同时 执行 ,ordering 属性 有 
sequentially 和 together 两 个 选项 ,其 中 together 为 默认 选项 ,sequentially 表示 动画 集合 (set) 中 的 多 
个 动画 ,按照 先后 顺序 执行 ,together 表示 动画 集合 (set) 中 的 多 个 动画 同时 执行 。< target android: 
name 一 "mytick"> 表 示 仅 对 mytick 执行 动画 。android:drawable 和 “QQdrawable/myvector" 用 于 指定 
矢量 图 形 文 件 myvector. xml。 矢 量 图 形 文件 myvector. xml 的 主要 代码 如 下 : 


<?xml version = "1.0" encoding = "utf 一 8"?> 

< Vector xmlns:android = "http://schemas. android. com/apk/res/android" 
android:width = "240dp" 
android: height = "240dp" 
android:viewportHeight = "24.0" 
android:viewportWidth = "24.0"> 

< group android: name = " mytick" 
android:pivotX= "12" 
android:pivotY = "12"> 
<path android: strokeColor = "#FFFF0000" 

android: strokeWidth= "1.5" 
android:pathData = "M 4,11 L 10,16 L 20 3"> 

</path></group ></vector > 
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上 面 这 段 代 码 在 MyCode\MySample613\app\src\main\res\drawable\myvector. xml 文件 中 。 
需要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile "com. android. support: 
appcompat-v7 :24. 0.0' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample613 文件 夹 中 。 


174 使 用 ViewPropertyAnimator 创建 多 个 动画 


此 实例 主要 通过 直接 在 ViewPropertyAnimator 上 创建 多 个 属性 动画 ,实现 在 图 像 控 件 上 同时 产 
生 多 种 动画 效果 。 当 实例 运行 之 后 , 单 击 “ 开 始 播放 动画 ?按钮 , 则 图 像 ( 妖 怪 ) 控 件 将 从 左上 角 沿 着 对 
角 线 方 回 平移 到 右 下 角 ,效果 分 别 如 图 174. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


开始 播放 动画 停止 播放 动画 开始 播放 动画 停止 播放 动画 


图 174. 1 


// 响 应 单 击 " 开 始 播放 动画 "按钮 
public void onClickButtonl (View v) { 
myViewPropertyAnimator = myImageView. animate( ); 
myViewPropertyAnimator. translationX(600) 
.translationY(1100). setDuration(5000). start( ); 


} 

// 响 应 单 击 " 停 止 播 放 动 画 " 按 钮 

public void onClickButton2(View v) { 
myViewPropertyAnimator. cancel( ); 


} 


上 面 这 上段 代码 在 MyCode\ MySamplel66\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myViewPropertyAnimator 二 mylmageView. animate() 
用 于 返回 ViewPropertyAnimator 对 象 。 myViewPropertyAnimator. translationX (600 ). 
translationY(1100). setDuration(5000). start() 表 示 在 5 秒 内 myImageView 控件 在 水 平方 回 上 移动 
600 ,同时 在 垂直 方向 上 移动 1100 ,合成 之 后 近似 于 屏幕 对 角 线 方向 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample166 文件 夹 中 ， 
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175 ” 自 定 义 selector 实现 以 动画 形式 改变 阴影 大 小 


此 实例 主要 通过 设置 CardView 控件 的 stateListAnimator 属性 为 自 定义 selector, 并 在 自 和 定义 
selector 中 使 用 objectAnimator 改变 translationZ 属性 值 , 实 现 以 动画 的 形式 改变 控件 的 阴影 大 小 。 
当 实 例 运行 之 后 , 单 击 CardView 控件 , 则 在 单 击 该 控件 时 增 大 该 控件 的 translationZ 属性 值 ,从 而 产 
生 上 浮 的 阴影 效果 ,如 图 175. 1 的 左 图 所 示 ; 在 离开 该 控件 时 减少 (人 恢复) 至 正常 状态 , 则 阴影 消失 ,如 
图 175. 1 的 右 图 所 示 。 
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图 175. 1 


< android. support. v7. widget. CardView 
xmlns:card view = "http://schemas. android. com/apk/res — auto" 
android: layout_width = "200dp" 
android:layout height = "wrap_content" 
android:layout centerInParent = "true" 
android:clickable = "true" 
android: stateListAnimator = "(Wanimator/myselector" 
card view:cardCornerRadius = "20dp" 
card view:cardPreventCornerOverlap = "true"> 
< ImageView android: layout width = "250dp" 
android:layout height = "350dp" 
android:background = "(@mipmap/myimagel"/> 
</android. support. v7. widget. CardView > 


上 面 这 段 代 码 在 MyCode\MySample575\app\src\main\res\layout\activity_main. xml 文件 中 。 
在 这 段 代 码 中 ,android:stateListAnimator 一 "animator/myselector" 用 于 设置 动态 改变 阴影 大 小 的 
目 定 义 选择 天 selector。 目 定义 选择 天 selector 的 主要 内 容 如 下 : 
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<?xml] version= "1.0" encoding = "utf 一 8"?> 
< selector xmlns:android = "http://schemas. android. com/apk/res/android"> 
< item android: state pressed = "true"> 
< Set > 
<objectAnimator android: duration = " (android: integer/config shortAnimTime" 
android: propertyName = " translationZ" 
android: valueTo = " 20dp" 
android: valueType = "floatType" /> </set ></item> 
< item> 
< Set > 
<objectAnimator android: duration = "(Oandroid: integer/config shortRnimTime" 
android: propertyName = "translationZ" 
android: valueTo = " 0dp" 
android: valueType = "floatType" /></set></itenm> 
</selector > 


上 和 面 这 段 代码 在 MyCode\MySample575\app\src\main\res\animator\myselector. xml 文件 中 。 
在 这 段 代 码 中 ,< item android: state_pressed 一 "true"> 表 示 在 控件 被 单 击 时 执行 此 标签 中 的 动画 。 
android:propertyName 一 "translationZ" 表示 改变 translationZ 属性 值 ,android: valueTo 王 "20dp" 表 
示 将 translationZ 属性 值 改 变 为 20dp。 需 要 说 明 的 是 ,如 果 当 前 项 目的 res 目录 下 不 存在 animator 
子 目 录 , 则 应 该 先 在 res 目录 下 创建 animator 子 目 录 , 然 后 创建 myselector. xml 文件 。 此 外 ,在 
Android Studio 中 ,使 用 此 实例 的 相关 控件 需要 在 gradle 中 引入 compile 'com. android. support: 
cardview 一 v7: 25. 3.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\MySample575 文件 夹 中 ， 


176 ”使 用 ripple 标签 创建 中 心 波 纹 扩散 动画 


此 实例 主要 通过 在 XML 文件 中 使 用 ripple 标签 创建 指定 颜色 和 形状 的 中 心 波纹 扩散 动画 并 设 
置 为 控件 的 背景 ,实现 在 单 击 控件 时 产生 波纹 扩散 效果 。 当 实例 运行 之 后 , 单 击 ImageView 控件 (地 
球 图 像 ) , 则 在 该 控件 上 产生 蓝 色 的 中 心 波纹 扩散 的 动画 ,效果 分 别 如 图 176. 1 的 左 图 和 右 图 所 示 。 
Q 中 局 上 午 11:34 | QQ bp 加 重 "AD 上 午 11:33 


| pw 


上 和 二 11:33 


MySample MySample 


图 176.1 


依 Android 炫 酷 应 用 300 例 。 实战 篇 


主要 代码 如 下 : 


// 响 应 单 击 图 像 产生 波纹 扩散 的 效果 
public void onClickImageView(View myView) { 
myView. setBackground(getDrawable(R. drawable. myripple) ); 


} 


上 面 这 段 代 码 在 MyCode\ MySample565\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myView. setBackground (getDrawable (R. drawable. 
myripple)) 表 示 设 置 动 画 资源 文件 myripple 作为 控件 myView 的 背景 。 动 画 资源 文件 myripple 的 主 


要 内 容 如 下 : 


<?xml version = "1.0”encoding = "utf - 8" ?> 
< ripple xmlns:android = "http://schemas. android. com/apk/res/android" 
android: color = "?android:attr/colorControlHighlight"> 
< item> 
< inset android: insetBottom = "16dp" 
android: insetLeft = " 14dp" 
android: insetRight = "14dp" 
android: insetTop = "16dp"> 
< Shape android: shape = "rectangle"> 
< corners android:radius ="150dp" /> 
< solid android:color = " 井 D1EEEE" /> 
< padding android: bottom = "14dp" 
android: left = "18dp" 
android: right = "18dp" 
android:top ="14dp" /> 
</shape ></ inset ></item></ripple> 


上 F 面 这 段 代 码 在 MyCode\MySample565\app\src\main\res\drawable\myripple. xml 文件 中 。 
此 实例 的 完整 项 目 在 MyCode\MySample565 文件 夹 中 。 


177 使 用 GLSurfaceView 实现 3D 地 球 的 自转 


此 实例 主要 通过 自 定义 GLSurfaceView 控件 的 泻 染 器 Renderer, 实 现 3D 地 球 的 自转 动画 。 当 
实例 运行 之 后 , 单 击 “开始 自转 ?按钮 , 则 3D 地 球 将 围绕 > 轴 顺 时 针 旋 转 ; 单 击 “ 停 止 自转 ”按钮 , 则 
3D 地 球 停止 自转 ,效果 分 别 如 图 177. 1 的 左 图 和 右 图 所 示 ，。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
GLRenderer myGLRenderer; 


(Override 
Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
GLSurfaceView myGLSurfaceView = 
(GLSurfaceView) findViewById(R. id.myGLSurface); 


Bitmap myBitmap = 
BitmapFactory. decodeResource(getResources(), R.mipmap.myearth); 
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myGLRenderer = new GLRenderer(myBitmap); 
myGLSurfaceView. setRenderer(myGLRenderer); 
myGLRenderer. isRotationEnabled = false; 


} 

// 响 应 单 击 " 开 始 自转 "按钮 

public void onClickBtnl (View v) { myGLRenderer. isRotationEnabled = true;} 

// 响 应 单 击 "停止 自转 "按钮 

public void onClickBtn2(View v) { myGLRenderer. isRotationEnabled = false;} 
} 
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图 177.1 


上 面 这 段 代 码 在 MyCode\ MySample779\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,myGLRenderer 二 new GLRenderer (myBitmap) 中 的 
GLRenderer 是 实现 了 GLSurfaceView. Renderer 接口 的 自 定 义 类 ,关于 该 自 定义 类 的 详细 内 容 请 参 
考 源 代码 中 的 MyCode\MySample779\app\src\main\java\com\bin\luo\ mysample\GLRenderer. 
java 文件 。 此 实例 的 完整 项 目 在 MyCode\MySample779 文件 夹 中 ，。 


音频 和 视频 


178 ”使 用 MediaPlayer 播放 本 地 mp3 音乐 文件 


此 实例 主要 通过 使 用 MediaPlayer ,实现 播放 SD 卡 上 的 mp3 音乐 文件 。 当 实例 运行 之 后 ,在 “ 音 
乐 文件 : ”输入 框 中 输入 在 SD 卡 上 存储 的 音乐 文件 全 路 径 , 如 “/storage /sdcard0/mymusicl. mp3”， 
然后 单 击 “开始 播放 ”按钮 , 则 将 播放 指定 的 音乐 文件 ,效果 分 别 如 图 178. 1 的 左 图 和 右 图 所 示 。 


MySample 


音乐 文件 :lcard0/mymusic1.mp3 ”开始 播放 | 习 


图 178.1 


public void onClickmyBtnl(View v) { // 响 应 单 击 " 开 始 播放 "按钮 

try { 
MediaPlayer myMediaPlayer = new MediaPlayer( ); 
myMediaPlayer. setDataSource(myEditText.getText().toString()); // 音 乐 文件 路 径 
myMediaPlayer. prepare( ); 
myMediaPlayer. start(); // 播 放 音 乐 

} catch (Exception e) { // 注 意 :输入 框 的 音乐 文件 必须 在 SD 

卡 中 真实 存在 
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Toast. makeText (getApplicationContext( ), 
e.getMessage(), Toast.LENGTH SHORT). show( ) ; 
大 


上 F 面 这 段 代 码 在 MyCode\ MySample259\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myMediaPlayer. setDataSource(myEditText. getText(). 
toString()) 用 于 设置 MediaPlayer 将 要 播放 的 音乐 文件 ,myMediaPlayer. start() 用 于 播放 音乐 。 此 
外 , 读 取 在 SD 卡 的 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android:name 王 " 
android. permission. READ_EXTERNAL _ STORAGE"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample259 文件 夹 中 。 


179 ”使 用 MediaPlayer 播放 本 地 mp4 视频 文件 


此 实例 主要 通过 使 用 MediaPlayer 和 SurfaceView ,实现 在 SD 卡 上 播放 mp4 视频 文件 。 当 实例 
运行 之 后 ,在 “视频 文件 : ”输入 框 中 输入 SD 卡 上 存储 的 视频 文件 全 路 径 , 然 后 单 击 “ 开 始 播 放 ” 按 钮 ， 
则 将 播放 指定 的 视频 文件 ,如 图 179. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


视频 文件 :dcard0/myvideo1.mp4 ”开始 播放 | 避 频 文件 :dcard0/myvideo2.mp4 开始 播放 


— 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

EditText myEditText; 

private SurfaceView mySurfaceView; 

private SurfaceHolder mySurfaceHolder; 

private MediaPlayer myMediaPlayer = null; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 


访 Android 炫 酷 应 用 300 例 。 实战 篇 


myEditText = (EditText)findViewById(R. id.myEditText); 
myEditText. setText(Environment. getExternalStorageDirectory( ) 

+ "/myvideol. mp4"); 
myMediaPlayer = new MediaPlayer( ); 
mySurfaceView = (SurfaceView) findViewById(R. id.mySurfaceView); 
mySurfaceHolder = mySurfaceView.getHolder( ); 
mySurfaceHolder. setFixedSize( 400, 800); 


} 
public void onClickmyBtnl (View v) { // 响 应 单 击 "开始 播放 "按钮 
try { 
myMediaPlayer. setDataSource(myEditText.getText().toString()); // 设 置 数 据 资 源 
myMediaPlayer. setDisplay(mySurfaceHolder); // 设 置 视频 播放 窗口 
myMediaPlayer. prepare( ); 
myMediaPlayer. start(); // 播 放 视频 
} 
catch ( IOException e) { // 注 意 : 输 入 框 的 视频 文件 必须 在 SD 


卡 中 真实 存在 
Toast. makeText (getApplicationContext( ), 
e. getMessage( ) Toast. LENGTH_ SHORT). show( ); 


大 加 | 


上 面 这 上段 代码 在 MyCode\ MySample260\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,myMediaPlayer. setDataSource(myEditText. getText(). 
toString ( )) 用 于 设置 MediaPlayer 将 要 播放 的 视频 文 件 。myMediaPlayer. setDisplay 
(mySurfaceHolder) 用 于 设置 视频 播放 窗口 。MediaPlayer 通常 用 于 播放 音频 ,但 是 如 果 需 要 使 用 
MediaPlayer 播放 视频 , 则 需要 使 用 SurfaceView 设置 视频 显示 窗口 。SurfaceView 比 普通 的 自 定义 
View 更 有 绘图 上 的 优势 , 它 支持 完全 的 OpenGL ES 库 。 此 外 , 读 取 在 SD 卡 上 的 视频 文件 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 二 "android. permission. READ_ 


EXTERNAL_STORAGE"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample260 文件 夹 中 。 


180 ”使 用 MediaPlayer 播放 指定 网 址 的 音乐 文件 


此 实例 主要 通过 使 用 MediaPlayer 的 setOnBufferingUpdateListener() 方 法 和 setDataSource() 
方法 ,实现 缓冲 并 播放 指定 网 址 的 网 络 音 乐 文件 。 当 实例 运行 之 后 ,在 "歌曲 网 址 : "输入 框 中 输入 音 
乐 文 件 的 网 络 地 址 ,然后 单 击 “ 下 载 并 播放 此 网 络 歌曲 ”按钮 , 则 将 立即 缓冲 网 络 数 据 , 完 成 之 后 则 播 
放 此 歌曲 文件 ; 单 击 右 边 的 暂停 按钮 , 则 可 以 暂停 播放 ,效果 分 别 如 图 180. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity 
implements SeekBar. OnSeekBarChangeListener { 

EditText myEditText; 
private MediaPlayer myMediaPlayer; 
TextView myTotalTime; 
ImageView myPlayPause; 
SeekBar mySeekBar; 
Handler myHandler = new Handler() { 

(@Override 

public void handleMessage(Message msg) { 
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myCurrentTime. setText (getPlayerCurrentPositionToString(myMediaPlayer)); 
| 国王 
private TextView myCurrentTime; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myEditText = (EditText) findViewById(R. id.myEditText); 
myCurrentTime = (TextView) findViewById(R. id.myCurrentTime); 
myTotalTime = (TextView) findViewById(R. id.myTotalTime); 
myPlayPause = (ImageView) findViewById(R. id.myPlayPause); 
myPlayPause. setOnClickListener(new View. OnClickListener() { 
(Override 
public void onClick(View view) { 
if (myMediaPlayer == null) return; 
if (myMediaPlayer. isPlaying()) { // 如 果 正 在 播放 , 则 暂停 
myMediaPlayer. pause( ); 
myPlayPause. setImageResource(R. mipmap. myicon play); 
} else { // 如 果 已 经 暂停 , 则 播放 
myMediaPlayer. start( ); 
myPlayPause. setImageResource(R.mipmap. myicon pause); 
3 
mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
} 
(Override 
public void onProgressChanged( SeekBar seekBar, int i, boolean b) { } 
(Override 
public void onStartTrackingTouch( SeekBar seekBar) { } 
(Override 
public void onStopTrackingTouch( SeekBar seekBar) { 
myMediaPlayer. seekTo( seekBar. getProgress( ) ) ; 


} 


音频 和 视频 人 > 


public void onClickmyBtnl(View v) { // 响 应 单 击 "下 载 并 播放 此 网 络 歌曲 "按钮 


try { 
myMediaPlayer = new MediaPlayer() ; 
myMediaPlayer. setAudioStreamTypel( AudioManager. STREAM MUSIC); 
myMediaPlayer. setDataSource(myEditText. getText().toString( ) ) ; 
myMediaPlayer. prepare( ); 
myMediaPlayer. start( ); 
myMediaPlayer. setOnBufferingUpdateListener( 
new MediaPlayer. OnBufferingUpdateListener() { 
(Override 
public void onBufferingUpdate(MediaPlayer mediaPlayer, int i) { 
mySeekBar. setSecondaryProgress((i/ 100) * 
myMediaPlayer. getDuration( ) ) ; // 显 示 缓 冲 进度 


if (i == 100) { 
Toast. makeText (MainActivity. this, 
"音乐 已 下 载 (缓冲 ) 完 成 !"，Toast. LENGTH_SHORT) . show( ); 
} } }); 
mySeekBar. setMax(myMediaPlayer. getDuration( ) ); 
myTotalTime. setText (getPlayerDurationToString(myMediaPlayer)); 
myPlayPause. setImageResource(R. mipmap. myicon pause); 


人 2》 Anaroi 红 本 应 有 300 。 实战 篇 


mySeekBar. setOnSeekBarChangeListener(this ) ; 
final Timer myTimer = newTimer( ) ; 
myTimer. Schedule(new TimerTask() { 
(Override 
public void run() { // 显 示 播 放 进 度 
mySeekBar. setProgress(myMediaPlayer. getCurrentPosition( ) ) ; 
myHandler. sendEmptYyMessage(1); 
jl 0 10); 
} catch (Exception e) { } 
} 
public String getPlayerDurationToString(MediaPlayer player) { // 获 取 音 乐 时 长 
return player. getDuration()/1000/60 + ":" + (player. getDuration()/1000 % 60 < 10 ? "0" + player. 
getDuration()/1000 % 60 : player.getDuration()/1000 % 60); 
} 
public String getPlayerCurrentPositionToString(MediaPlayer player) { 
return player. getCurrentPosition()/1000/60 + ":" + (Player. getCurrentPosition() /1000% 60<10?"0" 
+ player. getCurrentPosition( )/1000 % 60 : player. getCurrentPosition() / 1000 % 60); 
// 获 取 歌 曲 当前 播放 位 置 
上 


MySample MySample 


歌曲 网 址 : http://www.170mv.com/kw/ 歌曲 网 址 : http://www.170mv.com/kw/ 
other.web.ri01.sycdn.kuwo.cn/ other.web.ri01.sycdn.kuwo.cn/ 
resource/ resource/ 
n3/29/38/2466852912.mp3 n3/29/38/2466852912.mp3 


下 载 并 播放 此 网 络 歌曲 下 载 并 播放 此 网 络 歌曲 


2:59 : 2:59 [® 


图 180.1 


上 面 这 上段 代码 在 MyCode\ MySample719\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 其 中 ,myMediaPlayer. setOnBufferingUpdateListener() 方 法 用 于 监听 组 
冲 来 自 网 络 的 音乐 文件 ,可 以 在 其 参数 中 获取 缓冲 进度 。myMediaPlayer. setDataSource 
(myEditText. getText(). toString()) 用 于 在 参数 中 设置 网 络 音 乐 文件 。setDataSource() 方 法 设置 播 
放 源 的 第 用 调用 形式 如 下 。 

(1) 播放 当前 应 用 的 资源 文件 。 如 果 把 “test. mp3” 音 乐 文件 放 在 res/raw/ 目 录 下 , 则 调用 形式 如 
下 : myMediaPlayer. setDataSource(this, Uri. parse("android. resource: //com. android. sim/" 十 R. 
raw. test))。com. android. sim 是 当前 应 用 包 和 名 , 即 Uri. parse ("android. resource:// 应 用 程序 包 
名 /" 十 R. raw. 音乐 文件 名 称 )。 如 果 把 “test. mp3” 音 乐 文件 放 在 assets 目录 下 , 则 调用 形式 如 下 : 
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AssetManager assetMg = this. getApplicationContext().getAssets( ); 

AssetFileDescriptor fileDescriptor = assetMg. openFd("test. mp3"); 

myMediaPlayer. setDataSource(fileDescriptor. getFileDescriptor( ), 
fileDescriptor. getStartOffset(), fileDescriptor. getLength( ) ) ; 


(2) 播放 存储 卡 音 乐 文件 ,如 myMediaPlayer. setDataSource("/mnt/sdcard/test. mp3") 。 

(3) 播放 远程 的 资源 文件 ,如 myMediaPlayer. setDataSource(Context, Uri. parse ("http:// xx "))。 

此 外 ,访问 网 络 需 要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 三 " 
android. permission. INTERNET"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample719 文件 夹 中 。 


181 使 用 请 块 同步 MediaPlayer 播放 音频 的 进度 


此 实例 主要 通过 使 用 SeekBar 和 MediaPlayer, 实 现在 播放 音乐 时 ,SeekBar 的 滑 块 始终 跟随 
MediaPlayer 的 音乐 播放 进度 。 当 实例 运行 之 后 , 单 击 “ 选 择 音 乐 文件 ”按钮 , 则 可 以 在 打开 的 窗口 中 
选择 音乐 文件 ,如 图 181. 1 的 左 图 所 示 ; 选择 的 音乐 文件 的 时 长 将 显示 在 右 端 ,如 “5: 6”, 即 5 分 6 
秒 ; 单 击 下 面 的 播放 (或 暂停 ) 图 标 , 则 可 以 播放 (或 暂停 ) 音 乐 ; 在 播放 音乐 时 ,SeekBar 的 滑 块 将 自动 

跟随 音乐 播放 进度 癌 右 滑动 ; 回 右 或 回 左 拖 动 滑 块 , 则 可 调节 音乐 播放 进度 ,如 图 181. 1 的 右 图 所 示 。 


选择 音乐 文件 


mymusic.mp3 OO 
2017 年 9 月 2..。 2.01 MB 


mymusic1.mp3 
1 月 21 日 11,70 MB 


mymusic2.mp3 
1 月 21 日 7.21 MB 


mymusic3.mp3 
1 月 21 日 10.92 MB 


mymusic4.mp3 
1 月 21 日 14.41 MB 


mymusic5.mp3 
1 月 21 日 4.79 MB 


图 181.1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements View. OnClickListener, 
SeekBar. OnSeekBarChangeListener { 

boolean isPlaying = false; 

private TextView myCurrentTime; 

private TextView myTotalTime; 

private ImageView myPlayPause; 

private MediaPlayer myPlayer:; 

private SeekBar mySeekBar; 
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Handler myHandler = new Handler() { 
人 Override 
public void handleMessage( Message msg) { // 显 示 当 前 播放 位 置 
myCurrentTime. setText (myPlayer. getCurrentPosition( )/1000/60 + ":" 
+ myPlayer.getCurrentPosition() / 1000 % 60); 

Pk 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
myCurrentTime = (TextView) findViewById(R. id.myCurrentTime); 
myTotalTime = (TextView) findViewById(R. id.myTotalTime); 
Button myBtnFile = (Button) findViewById(R. id.myBtnFile); 
myPlayPause = (ImageView) findViewById(R. id.myPlayPause); 
myBtnFile. setOnClickListener(this); 
myPlayPause. setOnClickListener(this); 
mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
mySeekBar. setOnSeekBarChangeListener(this); 

} 


(Override 
public void onClick(View view) { 
if (view.getId() == R. id.myBtnFile) { // 响 应 单 击 " 选 择 音 乐 文件 "按钮 


Intent myIntent = new Intent(Intent.ACTION GET CONTENT); 
myIntent. setType("audio/ *x "); 
CT 
startActivityForResult( Intent. createChooser(myIntent, null), 2); 
} catch (android. content. ActivityNotFoundException ex) { } 
} else if (view.getId() == R. id.myPlayPause) { // 单 击 播放 或 暂停 图 标 
setPlayerStatus(! isPlaying); 
} } 
public void setPlayerStatus(boolean isplay) { 
if (isplay) { // 播 放 
myPlayPause. setImageResource(R. mipmap. myicon pause); 
isPlaying = true; 
myPlayer. start( ); 
} else { // 暂 停 
myPlayPause. setImageResource(R. mipmap. myicon play); 
isPlaying = false; 
myPlayer. pausel( ); 
} } 
public String getPath(Context context, Uri uri) { 
if ("content".equalsIgnoreCase(uri.getScheme())) { 
String[ ] myProjection = {"_ data"}; 
Cursor myCursor = null; 
try { 
myCursor = context.getContentResolver().query(uri, 
myProjection, null, null, null); 
int myIndex = myCursor. getColumnIndexOrThrow(” data" ); 
if (myCursor. moveToF 1irst( ) ) { return myCursor. getString(myIndex); } 
} catch (Exception e) { } 
} else if ("file".equalsIgnoreCase(uri. getScheme( ) ) ) { return uri. getPath(); } 
return null; 


} 
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(Override 
protected void onActivityResult( int requestCode, int resultCode, Intent data) { 
if (requestCode == 2) { 
if (resultCode == RESULT OK) { 
Uri myUri = data. getData( ) ; 
String myPath = getPath(MainActivity. this, myUri); 
myPlayer = new MediaPlayer( ) ; 
try { 
myPlayer. setDataSource( myPath); 
myPlayer. prepare( ); // 加 载 译 乐 文件 资源 
myTotalTime. setText(myPlayer. getDuration() / 1000 /60 + ":" 
+ myPlayer.getDuration() / 1000 % 60); 
setPlayerStatus(true); 
mySeekBar. setMax( myPlayer. getDuration( ) ) ; 
new Timer().schedule(new TimerTask() { 
(Override 
public void run() { // 根 据 播放 进度 设置 SeekBar 位 置 
mySeekBar. setProgress(myPlayer. getCurrentPosition( ) ) ; 
myHandler. sendEmptyMessage( 1); 
} }, 0, 10); 
} catch (Exception e) { } 
Tr} 
(Override 
public void onProgressChanged( SeekBar seekBar, int i, boolean b) { } 
(Override 
public void onStartTrackingTouch( SeekBar seekBar) { } 
(Override 
public void onStopTrackingTouch(SeekBar seekBar) { 
myPlayer. seekTo(seekBar. getProgress() ); // 根 据 SeekBar 位 置 设 置 播放 进度 
}} 


上 面 这 上段 代码 在 MyCode\ MySample712\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySeekBar. setMax(myPlayer. getDuration()) 用 于 根据 
音乐 文件 的 时 长 设置 SeekBar 的 最 大 值 。mySeekBar. setProgress(myPlayer. getCurrentPosition() ) 用 
于 根据 当前 音乐 播放 进度 设置 SeekBar 滑 块 的 当前 位 置 。myPlayer. seekTo(seekBar. getProgress() ) 
用 于 根据 SeekBar 滑 块 的 当前 位 置 ( 在 被 拖 动 之 后 ) 设 置 ( 重 置 ) 音 乐 播放 进度 。 此 外 , 读 取 在 SD 卡 上 
的 音乐 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 二 "android. 
permission. READ_EXTERNAL STORAGE"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample712 
文件 夹 中 。 


182 ”使 用 请 块 同步 MediaPlayer 播放 视频 的 进度 


此 实例 主要 通过 使 用 SeekBar 和 MediaPlayer, 实 现在 播放 视频 时 ,SeekBar 的 滑 块 始终 跟随 
MediaPlayer 的 视频 播放 进度 。 当 实例 运行 之 后 , 单 击 “选择 视频 文件 ”按钮 , 则 可 以 在 打开 的 窗口 中 
选择 视频 文件 ,如 图 182. 1 的 左 图 所 示 ; 选择 的 视频 文件 的 时 长 将 显示 在 右 端 ,如 “5: 53”, 即 5 分 53 
秒 ; 单 击 右 端 的 播放 (或 暂停 ) 图 标 , 则 可 以 播放 (或 暂停 ) 视 频 ; 在 播放 视频 时 ,SeekBar 的 滑 块 将 自动 
跟随 视频 播放 进度 回 右 滑动 ; 回 右 或 回 左 拖 动 滑 块 , 则 可 调节 视频 播放 进度 ,如 图 182. 1 的 右 图 所 示 。 
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Dh ds 


myvideo.mp4 


2017 年 9 月 1 39.49 MB 


图 182.1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements View. OnClickListener { 
boolean isPlaying = false; 
private SurfaceView mySurfaceView; 
private MediaPlayer myPlayer; 
private TextView myTotalTime; 
private ImageView myPlayPause; 
Handler myHandler = new Handler() { // 更 新 当前 播放 位 置 
(Override 
public void handleMessage(Message msg) { 
if (msg.what == 2) { 
String myCurrentPosition; 
if (myPlayer. getCurrentPosition() / 1000 % 60<10){ 
myCurrentPosition = "0" + myPlayer.getCurrentPosition() / 1000 % 60; 
} else { myCurrentPosition = myPlayer.getCurrentPosition()/1000%60 + "";} 
myCurrentTime. setText (myPlayer. getCurrentPosition() / 1000 /60 + ":" 
+ myCurrentPosition); 
} } }; 
private TextView myCurrentTime; 
private SeekBar mySeekBar; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
Button myBtnFile = (Button) findViewById(R. id.myBtnFile); 
myBtnFile. setOnClickListener(this); 
myPlayPause = (ImageView) findViewById(R. id.myPlayPause); 
mySurfaceView = (SurfaceView) findViewById(R. id. mySurfaceView) ; 
mySeekBar = (SeekBar) findViewById(R. id. mySeekBar ) ; 
myCurrentTime = (TextView) findViewById(R. id. myCurrentTime) ; 
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myTotalTime = (TextView) findViewById(R. id.myTotalTime); 
} 
(Override 
public void onClick(View view) { // 响 应 单 击 "选择 视频 文件 "按钮 
if (view. getId() == R. id.myBtnFile) { 
Intent myIntent = new Intent(Intent.RCTION GET CONTENT ) ; 
myIntent. setType("video/ * "); // 选 择 视频 类 型 文件 
myIntent.addCategory( Intent. CATEGORY OPENABLE); 
try { 
startActivityForResult( Intent. createChooser(myIntent, null), 1); 
} catch (Exception ex) { } 
时 
// 响 应 单 击 播放 (或 暂停 ) 按 钮 ( ImageView 控件 ) 
public void onClickPlayPause(View view) { setPlayerStatus(!isPlaying); } 
public void setPlayerStatus(boolean isplay) { 
if (isplay) { // 播 放 
myPlayPause. setImageResource(R. mipmap. myicon pause); 
isPlaying = true; 
myPlayer. start( ); 
}else{ // 暂 停 
myPlayPause. setImageResource(R. mipmap.myicon play); 
isPlaying = false; 
myPlayer. pause( ); 
}} 
(Override 
protected void onActivityResult(int requestCode, int resultCode, Intent data){ 
if (requestCode == 1){ 
if (resultCode == RESULT OK){ 
Uri myUri = data.getData( ); 
final String myPath = getPath(this, myUri); 
try { 
mySurfaceView. getHolder( ). setKeepScreenOn(true); 
mySurfaceView. getHolder( ).addCallback(new SurfaceHolder.Callback( ){ 
(Override 
public void surfaceCreated( SurfaceHolder surfaceHolder){ 
try { 
myPlayer = new MediaPlayer( ); 
myPlayer. setDataSource( myPath); 
myPlayer. setDisplay(mySurfaceView. getHolder( ) ); 
myPlayer. Prepare( ); 
setPlayerStatus(true); 
new Timer().schedule(new TimerTask( ){ 
(Override 
public void run(){ // 根 据 播放 进度 设置 SeekBar 的 当前 值 
mySeekBar. setProgress(myPlayer. getCurrentPosition( ) ) ; 
myHandler. sendEmptyMessage(2); 
} }, 0, 10); 
myTotalTime. setText (myPlayer. getDuration() / 1000 / 60 + ":" 
+ myPlayer.getDuration() / 1000 % 60); 
// 根 据 视 频 文件 的 时 长 设置 SeekBar 控件 的 最 大 值 
mySeekBar. setMax(myPlayer. getDuration( ) ); 
mySeekBar. setOnSeekBarChangeListener!( 
new SeekBar. OnSeekBarChangeListener() { 
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(Override 
public void onProgressChanged(SeekBar seekBar，int i, boolean b) {} 
(Override 
public void onStartTrackingTouch( SeekBar seekBar) {} 
(Override 
public void onStopTrackingTouch(SeekBar seekBar) { 
// 根 据 SeekBar 控件 的 当前 值 设 置 播放 进度 
myPlayer. SeekTo( seekBar. getProgress( ) ); 
3 
} catch (IOException e) { e.printStackTrace( ); } 
} 
(Override 
public void surfaceChanged( SurfaceHolder surfaceHolder, 
qm 
(Override 
public void surfaceDestroyed( SurfaceHolder surfaceHolder) { } }); 
} catch (Exception e){ e.printStackTrace( ); } 
1 1) 
public static String getPath(Context context, Uri uri) { // 获 取 视 频 文件 全 路 径 
if ("content".equalsIgnoreCase(uri.getScheme())) { 
String[ ] projection = {"_ data"}; 
Cursor cursor = null; 
try { 
cursor = context. getContentResolver().query(uri, 
projection, null, null, null); 
int column index = cursor.getColumnIndexOrThrow(" data" ); 
if (cursor.moveToFirst()) { return cursor.getString(column index); } 
} catch (Exception e) { } 
} else if ("file".equalsIgnoreCase(uri. getScheme( ) ) ) { return uri. getPath( ); } 
return null; 


} } 


上 面 这 上段 代码 在 MyCode\ MySample715\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample715 文件 夹 中 。 


183 ”使 用 MediaController 创建 视频 播放 控制 栏 


此 实例 主要 通过 使 用 MediaController 对 象 作 为 VideoView 的 setMediaController () 方 法 的 参 
数 ,实现 在 单 击 VideoView 控件 之 后 ,在 屏幕 底部 弹出 视频 播放 控制 栏 。 当 实例 运行 之 后 , 单 击 “ 浏 览 
本 地 视频 文件 ”按钮 ,然后 在 弹出 的 窗口 中 选择 视频 文件 , 即 可 在 VideoView 控件 中 自动 播放 此 视频 ; 
单 击 正在 播放 的 视频 , 则 在 屏幕 底部 弹出 控制 视频 播放 的 控制 栏 MediaController, 在 此 控制 栏 上 有 播 
放 、 暂 停 等 按钮 ,以 及 播放 进度 条 , 单 击 相关 按钮 即 可 控制 视频 播放 ,效果 分 别 如 图 183. 1 的 左 图 和 碳 
图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
VideoView myVideoView; 
Button myBtnBrowser; 
(Override 
Protected void onCreate(Bundle savedInstanceState) { 
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super. onCreatel( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myBtnBrowser = (Button) findViewById(R. id.myBtnBrowser); 
myVideoView = (VideoView) findViewById(R. id.myVideoView); 
myBtnBrowser. setOnClickListener(new View. OnClickListener() { 
(Override 
public void onClick(View v) { // 浏 览 视 频 文 件 
Intent myIntent = new Intent(Intent.ACTION GET CONTENT ) ; 
myIntent. setTYpe(" video/mp4" ); 
startActivityForResult(myIntent, 0); 
} }); 
} 
(Override 
protected void onRctivityResult(int requestCode, int resultCode, Intent data) { 
if(resultCode == RESULT OK){ 


Uri myUri = data. getData( ); // 获 取 用 户 所 选 视频 Uri 
myVideoView. setVideoURI(myUri); // 设 置 Uri( 视 频 文件 全 路 径 ) 
myVideoView. start( ); // 播 放 视 频 


// 在 单 击 视频 之 后 ,显示 播放 、 暂 停 等 按钮 的 控制 栏 ,初始 化 MediaController 对 象 
MediaController myMediaController = new MediaController(this); 

// 将 该 对 象 应 用 于 VideoView 控件 

myVideoView. setMediaController(myMediaController); 


Ey 


MySample MySample 


浏览 本 地 视频 文件 浏览 本 地 视频 文件 
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上 面 这 段 代 码 在 MyCode\ MySample814 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myVideoView. setMediaController (myMediaController) 
用 于 实现 在 单 击 视频 之 后 ,在 屏幕 底部 显示 播放 控制 栏 。 此 实例 的 完整 项 目 在 MyCode\\ 
MySample814 文件 夹 中 。 
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184 ”使 用 MediaMetadataRetriever 实现 视频 截图 


此 实例 主要 通过 使 用 MediaMetadataRetriever 的 getFrameAtTime ( ) 和 MediaPlayer 的 
getCurrentPosition() 方 法 ,实现 对 当前 正在 播放 的 视频 进行 截图 。 当 实例 运行 之 后 ,在 "视频 文件 : ” 
输入 框 中 输入 在 SD 卡 上 存储 的 视频 文件 名 称 , 然 后 单 击 “ 开 始 播放 视频 "按钮 , 则 将 播放 此 视频 ,如 
图 184. 1 的 左 图 所 示 。 单 击 “ 当 前 视频 截图 ”按钮 , 则 将 把 视频 的 当前 图 像 保 存在 SD 卡 上 ,如 
图 184. 1 的 右 图 所 示 。 


dQmy 


MySample 


视频 文件 :/storage/sdcard0/myvideo.mp4 


1516931311445.jpg 


开始 播放 视频 当前 视频 截图 ee 大 小 70.07 KB 
1516931424355 jpg 
大 小 70.07 KB 


1516931548258.jpg 
大 小 70.07 KB 


1516931735258.jpg 
大 小 70.07 KB 


1516931917664.jpg 
大 小 45.49 KB 


1516932057963.jpg 
大 小 45.49 KB 


1516932115701.jpg 
大 小 45.49 KB 


1516932255434.jpg 
大 小 72.62 KB 


1516932312400.jpg 
大 小 52.68 KB 


主要 代码 如 下 : 


public void onClickmyBtn2(View v) { // 响 应 单 击 "当前 视频 截图 "按钮 
try { 
MediaMetadataRetriever myMediaMetadataRetriever = 
new MediaMetadataRetriever( ); 
myMediaMetadataRetriever. setDataSource(myVideoFile.getText().toString( )); 
Bitmap myBitmap = myMediaMetadataRetriever. getFrameAtTime (myMediaPlayer. getCurrentPosition() * 
1000, MediaMetadataRetriever.OPTION CLOSEST SYNC); 
if (myBitmap != null) { 
Uri myImageUri = getContentResolver(). insert(MediaStore. Images. Media. EXTERNAL CONTENT URI, new 
ContentValues( ) ); 
OutputStream myOutStream = 
getContentResolver().openOutputStream(myImageUri); 
myBitmap. compress(Bitmap. CompressFormat. JPEG, 
90, myOutStreanm); // 将 截图 保存 到 存储 卡 
Toast. makeText (MainActivity. this, 
"截图 已 成 功 保存 !", Toast. LENGTH SHORT). show( ); 
} 
} catch (Exception e) { e.printStackTrace(); } 
} 
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上 面 这 段 代码 在 MyCode\MySample681\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代码 中 ,myBitmap 三 myMediaMetadataRetriever. getFrameAtTime(myMediaPlayer. 
getCurrentPosition() * 1000, MediaMetadataRetriever. OPTION CLOSEST_SYNC) 用 于 将 当前 视频 播 
放 位 置 ( 帧 位 置 ) 的 图 像 转 化 成 位 图 ,myMediaPlaver. getCurrentPosition() 用 于 获取 当前 视频 的 播放 位 置 
( 帧 位 置 ) ,由 于 getFrameAtTime() 方 法 的 第 一 个 参数 代表 的 是 微 秒 值 ,getCurrentPosition() 方 法 的 返回 
值 是 毫秒 值 , 因 此 需要 乘 以 1000。 使 用 getFrameAtTime(long timeUs，int option) 方 法 还 需要 注意 的 
是 : option:OPTION_CLOSEST 表示 获取 离 timeUs 最 近 的 一 帧 图 像 , 此 种 情况 获取 的 帧 是 Sync 
Frame， 由 于 在 timeUs 此 时 间 点 不 一 定 恰 好 有 一 个 Sync Frame， 所 以 可 能 有 一 定 的 误差 ,在 获取 
Sync Frame 时 可 能 有 下 列 3 种 情况 。 

(1) 在 timeUs 处 恰好 有 一 个 Sync Frame。 

(2) 获取 timeUs 前 一 个 Sync Frame。 

(3) 获取 timeUs 后 一 个 Sync Frame。 

此 外 ,在 SD 卡 读 写 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: 
name 一 " android. permission. READ EXTERNAL STORAGE"/> 和 < uses-permission android: 
name 二 "android. permission. WRITE_EXTERNAL_STORAGE" 放 权限 。 此 实例 的 完整 项 目 在 
MyCode\MySample681 文件 夹 中 。 


185 ”使 用 MediaMetadataRetriever 获取 视 频 缩 略图 


此 实例 主要 通过 使 用 MediaMetadataRetriever 的 成 员 方 法 getFrameAtTime() ,实现 根据 指定 的 
视频 文件 获取 其 视频 缩 略图 。 当 实例 运行 之 后 ,在 “视频 文件 : ”输入 框 中 输入 在 SD 卡 上 存储 的 视频 
文件 名 称 , 然 后 单 击 “获取 此 视频 文件 的 缩 略 图 ”按钮 , 则 在 下 面 显示 此 视频 文件 的 缩 略 图 ,效果 分 别 
如 图 185. 1 的 左 图 和 右 图 所 示 。 


MySample 


视频 文件 : ”torage/sdcard0/myvideo1lmp4 上 视频 文件 :torage/sdcard0/myvideo2.mp4 


获取 此 视频 文件 的 缩 略图 获取 此 视频 文件 的 缩 略 图 


图 185.1 
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主要 代码 如 下 : 


public void onClickmyBtn1(View v) {// 响 应 单 击 "获取 此 视频 文件 的 缩 略图 "按钮 
String myPath = myEditText. getText( ) .toString( ) ; 
Bitmap myBitmap = createVideoThumbnail(myPath, 
MediaStore. Images. Thumbnails.MINI KIND); 
myImageView. setImageBitmap(myBitmap); 
} 
public Bitmap createVideoThumbnail(String myPath, int kind) { 
MediaMetadataRetriever myRetriever = new MediaMetadataRetriever( ); 
myRetriever. setDataSource(myPath); 
Bitmap myBmp = myRetriever.getFrameAtTime( - 1); 
if (myBmp == null) return null; 
if (kind == MediaStore. Images. Thumbnails.MINI KIND) { 
int width = myBmp. getWidth( ); 
int height = myBmp. getHeight( ); 
int max = Math.max(width, height); 
if (max > 512) { 
float scale = 512f / max; 
int w = Math.round(scale * width); 
int h = Math.round(scale * height); 
myBmp = Bitmap. createScaledBitmap(myBmp, w, h, true); 
} 
return myBmp; 


上 面 这 段 代 码 在 MyCode\ MySample611 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 , myRetriever. setDataSource (myPath) 用 于 设置 
MediaMetadataRetriever 操作 的 视频 文件 。myBmp 二 myRetriever. getFrameAtTime( 一 1) 用 于 获取 
该 视频 指定 时 间 点 的 图 像 。myBmp 二 Bitmap. createScaledBitmap(myBmp,， w，h, true) 用 于 根据 指 
定 尺 才 创 建 该 图 像 的 缩 略 图 。 此 外 , 读 取 在 SD 卡 上 的 文件 需要 在 AndroidManifest. xml 文件 中 添加 
< Uses-permission android:name 王 "androld. permission. READ_EXTERNAL STORAGE"/ > 权限 。 
此 实例 的 完整 项 目 在 MyCode\MySample611 文件 夹 中 。 


186 ”使 用 VideoView 播放 本 地 mp4 视频 文件 


此 实例 主要 通过 使 用 VideoView ,实现 在 SD 卡 上 播放 mp4 视频 文件 。 当 实例 运行 之 后 ,在 “ 视 
频 文件 : ”输入 框 中 输入 在 SD 卡 上 存储 的 视频 文件 路 径 , 然 后 单 击 “ 开 始 播 放 ” 按 钮 , 则 将 播放 指定 的 
视频 文件 ,效果 分 别 如 图 186. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickmyBtn1 (View v) { // 响 应 单 击 " 开 始 播放 "按钮 
myVideoView. setMediaController(new MediaController(this) ); 
Uri myUri = Uri.parse(myEditText. getText( ) .toString( ) ) ; 
myVideoView. setVideoURI(myUri); 
myVideoView. start( ); 
myVideoView. requestFocus( ); 
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MySample MySample 


视频 文件 :lcard0/myvideo2.mp4 ”开始 播放 


图 186.1 


上 面 这 段 代 码 在 MyCode\ MySample261 \app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myVideoView. setVideoURI(myUri) 用 于 以 Uri 的 方式 
设置 VideoView 控件 将 要 播放 的 视频 文件 ,也 可 以 直接 使 用 myVideoView. setVideoPath 
(myEditText. getText( ). toString ( )) 方 法 设置 VideoView 控件 的 视频 文件 。 然 后 就 可 以 调用 
VideoView 控件 的 start() 方 法 来 控制 视频 的 播放 。 由 于 VideoView 控件 通过 与 MediaController 类 
结合 使 用 ,因此 可 以 不 用 自己 控制 视频 的 播放 与 暂停 等 操作 。 此 外 , 读 取 在 SD 卡 上 的 视频 文件 需要 
在 AndroidManifest. xml 文件 中 添加 < uses-permission android:name 二 "android. permission. READ 
_EXTERNAL_STORAGE"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample261 文件 夹 中 。 


187 ”使 用 VideoView 播放 指定 网 址 的 视频 文件 


此 实例 主要 通过 使 用 VideoView 的 setVideoPath() 方 法 设置 视频 文件 的 网 络 地 址 ,实现 播放 存 
储 在 网 络 上 的 视频 文件 。 当 实例 运行 之 后 ,在 “视频 网 址 : ”输入 框 中 输入 在 网 络 上 存储 的 视频 文件 网 
址 (如 “https://1251412368. vod2. myqcloud. com/vodtransgzp 1251412368/4564972818539678338/ 
v. {30. mp4”) ,然后 单 击 “ 播 放 视 频 ” 按 钮 , 则 视频 播放 效果 如 图 187. 1 的 左 图 所 示 。 在 “视频 网 址 : ” 
输入 框 中 输入 在 网 络 上 存储 的 其 他 视频 文件 网 址 (如 “https://1251412368. vod2. myqcloud. com/ 
vodtransgzp1251412368/45649728 18871314502/v. {30. mp4”) ,然后 单 击 “ 播 放 视 频 ? 按 钮 , 则 视频 播 
放 效 果 如 图 187. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickBtnl (View v) { // 响 应 单 击 "播放 视频 "按钮 
// 初 始 化 视频 控制 块 
MediaController myController = new MediaController(this); 
// 将 视频 控制 块 与 VideoView 控件 绑 定 
myVideoView. setMediaController(myController); 
// 设 置 网 络 视频 的 地 址 
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myVideoView. setVideoPath(myEditText. getText( ) .toString( ) ) ; 
myVideoView. start( ) ; // 开 始 播放 
} 


MySample MySample 


视频 网 址 : https:// 视频 网 址 : https:// 
1251412368.vod2.myqcloud.co 1251412368.vod2.myqcloud.co 
m/ m/ 


vodtransgzp1251412368/45649 vodtransgzp1251412368/45649 
72818539678338/v.f30.mp4 72818871314502/v.f30.mp4 


播放 视频 


图 187.1 


上 面 这 段 代 码 在 MyCode\ MySample816\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 由 于 VideoView 控件 能 够 通过 与 MediaController 控件 结合 使 用 ,因此 可 
以 不 用 自己 控制 视频 的 播放 与 暂停 等 操作 。 需 要 说 明 的 是 ,播放 网 络 视频 需要 在 AndroidManifest. 
xml 文件 中 添加 < uses-permission android: name 王 "android. permission. INTERNET"/ 光 权限。 此 
外 ,此 实例 在 部 分 模拟 器 测试 不 成 功 ,因此 请 直接 在 手机 上 测试 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample816 文件 夹 中 ， 


188 ”使 用 MediaRecorder 了 录制 音频 文件 


此 实例 主要 通过 使 用 录音 机 对 象 MediaRecorder, 实 现 使 用 手机 录制 音频 文件 。 当 实例 运行 之 
后 , 单 击 “ 开 始 录 音 ” 按 钮 , 则 实例 自动 根据 当前 时 间 生 成 一 个 文件 记录 音频 数据 ,此 时 就 可 以 对 着 手 
机 讲话 ,内 容 自 然 就 保存 在 此 音频 文件 中 ,如 图 188. 1 的 左 图 所 示 。 单 击 “ 停 止 录音 ”按钮 , 则 音频 文 
件 录制 完成 ,此 时 即 可 以 在 SD 卡 找 到 此 文件 ,然后 进行 播放 ,如 图 188. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickmyBtnl (View v) { // 响 应 单 击 "开始 录音 "按钮 
myFile = new File(Environment.getExternalStorageDirectory() + "/" + 
"myMediaRecorder" + new DateFormat().format("yyyyMMdd hhmnmss", 
Calendar. getInstance(Locale. CHINA)) + " .amr" ); // 将 录音 保存 在 SD 卡 


myMediaRecorder = new MediaRecorder(); // 创 建 录音 对 象 
myMediaRecorder. setAudioSource( 
MediaRecorder. AudioSource. DEFAULT) ; // 从 话 简 源 进行 录音 


myMediaRecorder. setOutputFormat( 
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MediaRecorder. OutputFormat. DEFRULT) ; // 设 置 输出 格式 
myMediaRecorder. setAudioEncoder( 

MediaRecorder. AudioEncoder. DEFAULT); // 设置 编码 格式 
myMediaRecorder. setOutputFile(myFile. getAbsolutePath( ) ); // 设置 输出 文件 
try { 

myMediaRecorder. prepare( ); // 准 备 录制 
myMediaRecorder. start(); // 开 始 录 制 
myTextView. setText(myFile. getAbsolutePath() + "文件 正在 录制 中 … … wk 


} catch (IllegalStateException e) { e.printStackTrace( ); } 
catch (IOException e) { e.printStackTrace( ); } 


} 
public void onClickmyBtn2(View v) { // 响 应 单 击 "停止 录音 "按钮 
if (myMediaRecorder != null) { // 如 果 正 在 录音 , 则 可 以 停止 录音 
myMediaRecorder. stop( ); 
myMediaRecorder. release( ); 


myMediaRecorder = null; 
myTextView. setText(myFile.getAbsolutePath() + "文件 录制 完毕 " ); 
}) 


MySample 


开始 录音 停止 录音 开始 录音 停止 录音 


storage/sdcard0/ storage/sdcard0/ 
yRecorder20170923_030650.amr 文 件 正在 lmyRecorder20170923_030650.amr 文 件 录制 
RR 制 中 毕 


pL 


图 188.1 


上 面 这 段 代 码 在 MyCode\ MySample268\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myMedia Recorder 二 new MediaRecorder() 用 于 创建 一 
个 录音 机 对 象 。myMedia Recorder. setOutputFile(myFile. getAbsolutePath()) 用 于 设置 音频 输出 文 
件 。myMedia Recorder. prepare() 用 于 对 录音 机 进行 预 处 理 , 如 果 没 有 此 操作 ,运行 可 能 会 报错 。 
myMedia Recorder. start() 表 示 开 始 录 制 音频 。myMedia Recorder. stop() 表 示 停 止 音 频 录 制 。 此 
外 ,在 SD 卡 写 入 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android:name 一 " 
android. permission. RECORD _ AUDIO"/> 和 < uses-permission android: name 一 " android. 
permission. WRITE _ EXTERNAL _ STORAGE "/> 权限 。 此 实例 的 完整 项 目 在 MyCodeA\ 
MySample268 文件 夹 中 。 
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189 ”使 用 RemoteViews 在 通知 栏 上 创建 播放 费 


此 实例 主要 通过 使 用 registerReceiver() 方 法 ,实现 以 注册 广播 服务 的 方式 在 通知 栏 上 创建 播放 
做 播放 音乐 。 当 实例 运行 之 后 , 单 击 "选择 音乐 文件 ?按钮 , 则 可 以 在 打开 的 窗口 中 选择 音乐 文件 ,如 


“小 品 : 梦幻 家 园 _ 超 清 . mp3”, 如 图 189. 1 的 左 图 所 示 ， 


同时 自动 播放 此 音乐 文件 ; 下 拉 滑 出 通知 栏 ， 


则 可 以 发 现 通知 栏 列表 项 有 播放 天 正在 播放 "小 品 : 梦幻 家 园 _ 超 清 . mp3”, 如 图 189. 1 的 右 图 所 示 ; 


单 击 右 侧 的 按钮 , 则 可 以 播放 或 暂停 。 


已 抓 取 屏 幕 截图 。 


“© 触摸 可 查看 您 的 屏幕 截图 。 


给 


歌曲 《光辉 岁月 》 … 合 大 鹏 _ 超 清 .mp3 
1 月 5 日 5.74 MB 


小 品 : 梦幻 家 园 超 清 .mp3 
1 月 5 日 15.45 MB 


alarm01.wav 
2017 年 10 月 .， 480 KB 


alarm02.wav 
2017 年 10 月 .， 323 KB 


图 189.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

boolean isPlaying = false; 

private MediaPlayer myMediaPlayer; 

private RemoteViews myRemoteViews; 

private NotificationManager myNotificationManager; 

private Notification myNotification; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 


myNotificationManager = 


小 品 : 梦幻 家 园 _ 超 清 .mp3 © 


已 作为 媒体 设备 连接 
触摸 可 显示 其 他 USB 选 项 。 


已 连接 到 USB 调 试 
触摸 可 停 用 USB 调 试 。 


// 播 放 暂 停 状 态 判 断 标志 


// 通 知 对 应 的 自 定义 视图 对 象 
// 通 知 管理 器 对 象 
// 通 知 对 象 


(NotificationManager) getSystemService( NOTIFICATION SERVICE ) ; 
myNotification = new Notification(R. drawable. myplayer., 
null, System. currentTimeMillis( )); 


// 自 定义 通知 栏 播 放 器 布局 
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myRemoteViews = new RemoteViews(getPackageName(), R. layout.myplayerlayout); 
myNotification. contentView = myRemoteViews; 
registerReceiver(new BroadcastReceiver() { 
@ Override 
public void onReceive(Context context, Intent intent) { 
if (intent.getAction() == "SwitchStatus") { 
if (isPlaying) { // 如 果 正 在 播放 , 则 暂停 
myMediaPlayer. pause( ) ; 
isPlaying = false; 
myRemoteViews. setImageViewResource(R. id.myBtnPlayPause, 
R. mipmap. myicon play); 
myNotificationManager. notify(1, myNotification); 
} else { // 如 果 已 经 暂停 , 则 播放 
myMediaPlayer. start( ); 
isPlaying = true; 
myRemoteViews. setImageViewResource(R. id. myBtnPlayPause, 
R. mipmap. myicon pause); 
myNotificationManager. notify(1, myNotification); // 更 新 并 重新 推送 通知 
} } } }, new IntentFilter("SwitchStatus" )); // 注 册 广 播 , 单 击 处 理 通 知 按钮 事件 
// 新 建 Intent, 用 于 接收 广播 时 过 滤 Intent 信息 
Intent myIntentSwitchStatus = new Intent("SwitchStatus" ); 
PendingIntent myPendingIntent = 
PendingIntent.getBroadcast(this, 0,myIntentSwitchStatus, 0); 
myRemoteViews. setOnClickPendingIntent(R. id.myBtnPlayPause, myPendingIntent ); 
myNotification.flags = myNotification. FLAG NO CLEAR; 
myNotificationManager. notify(1, myNotification); 
Button myBtnFile = (Button) findViewById(R. id.myBtnFile); 
// 浏 览 音 乐 文件 并 播放 
myBtnFile. setOnClickListener(new View. OnClickListener() { 
(@ Override 
public void onClick(View view) { 
Intent myIntent = new Intent(Intent.ACTION GET CONTENT); 
myIntent. setType("audio/ x* "); 
try { 
startActivityForResult(Intent. createChooser(myIntent, null), 31415926); 
} catch (android. content. ActivityNotFoundException ex) { } } }); 
} 
(Override 
protected void onActivityResult(int requestCode, int resultCode, Intent data){ 
if (requestCode == 31415926) { 
if (resultCode == RESULT OK) { 
Uri myUri = data. getData( ); 
String myPath = getFilePath(MainActivity.this, myUri); 
myMediaPlayer = new MediaPlayer( ); 
try { 
myMediaPlayer. setDataSource( myPath); 
myMediaPlayer. prepare( ); 
// 在 通知 栏 自 定义 播放 器 上 显示 音乐 文件 名 称 
myRemoteViews. setTextViewText(R. id.myFileName, 
getFileName(MainActivity.this, myPath)); 
myNotificationManager. notify(1, myNotification); 
myMediaPlayer. start( ); // 开 始 播放 
isPlaying = true; 
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} catch (Exception e) { } 
3 


// 获 取 音 乐 文件 的 完整 路 径 , 省略 的 部 分 请 参考 源 代码 
public String getFilePath(Context context, Uri uri) { } 


// 获 取 音 乐 文件 的 显示 名 称 ,省 略 的 部 分 请 参考 源 代码 
public static String getFileName(Context context, String path) { } 


上 面 这 上段 代码 在 MyCode\ MySample713\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,myRemoteViews 三 new RemoteViews (getPackageName()， 
R. layout. myplayerlayout) 用 于 根据 布局 文件 myplayerlayout. xml 在 通知 栏 上 创建 音乐 播放 器 。 关 
于 myplayerlayout 布局 的 详细 内 容 请 参考 源 代码 中 的 MyCode\MySample713\app\src\main\res\ 
layout\myplayerlayout. xml 文件 。 此 外 ,该 取 SD 卡 文件 需要 在 AndroidManifest. xml 文件 中 添加 
< Uses-permission android: name= "android. permission. READ EXTERNAL STORAGE"/ > 权限 。 
此 实例 的 完整 项 目 在 MyCode\MySample713 文件 夹 中 。 


190 ”在 使 用 SurfaceView 播放 视频 时 实现 横 屏 显示 


此 实例 主要 通过 使 用 setRequestedOrientation() 方 法 ,实现 在 播放 视频 时 进行 槛 屏 和 紧 屏 切换 。 
当 实 例 运行 之 后 , 单 击 “ 选 择 视 频 文件 ”按钮 , 则 可 以 在 打开 的 窗口 中 选择 视频 文件 ,选择 的 视频 文件 
的 时 长 将 显示 在 右 端 ,如 “5: 53”, 即 5 分 53 秒 ,并 且 自 动 开 始 播 放 ; 单 击 “ 横 屏 竖 屏 切 换 ” 按 钮 ,如 果 
当前 手机 是 竖 屏 , 则 切换 为 模 屏 ; 如 果 当 前 手机 是 横 屏 , 则 切换 为 竖 屏 ,效果 分 别 如 图 190. 1 的 左 图 和 
右 图 所 示 。 
万 WO 全 “dd 99% 上 午 11:26 


选择 视频 文件 模 屏 竖 屏 切换 


寺 OOB 趟 DD” 


全 


材 闪 泛 嘱 在 允 


芋 
局 
版 
油 
和 
落 


用 
ES 
DD 
8 
了 
二 
a 
已 
休 


图 190. 1 
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主要 代码 如 下 : 


public class MainActivity extends Activity implements SurfaceHolder. Callback { 
private MyUtils myUtils; 
private SurfaceView mySurfaceView; 
private SeekBar mySeekBar; 
Handler myHandler = new Handler() { 
人 Override 
public void handleMessage(Message msg) { // 显 示 已 播放 时 间 
myCurrentTime. setText(myUtils. getMediaPlayerCurrentPositionByString( ) ) ; 
最 于 
private TextView myCurrentTime, myTotalTime; 
private String myFilePath; 
(Override 
public void onConfigurationChanged(Configuration newConfig) { 
super. onConfigurationChanged(newConfig); 
if (newConfig. orientation == Configuration.ORIENTATION LANDSCAPE) { 
setVideoParams(true); 
} else if (newConfig.orientation == Configuration.ORIENTATION PORTRAIT){ 
setVideoParams(false); 
} } 
public void setVideoParams(boolean isLandscape) { 
LinearLayout myMainLayout = (LinearLayout) findViewById(R. id.activity main); 
ViewGroup. LayoutParams myMainLayoutParams = myMainLayout. getLayoutParams( ); 
ViewGroup. LayoutParams mySurfaceViewParams = mySurfaceView.getLayoutParams( ); 
float myScreenWidth = getResources().getDisplayMetrics().widthPixels; 
float myScreenHeight = getResources().getDisplayMetrics().heightPixels; 
if (isLandscape) { // 横 屏 显示 模式 
//getWindow( ) .addFlags(WindowManager. LayoutParams. FLAG FULLSCREEN); 
// 根 据 视频 原始 宽度 和 高 度 设置 播放 器 的 宽度 和 高 度 
mySurfaceViewParams.width = myUtils.getMyPlayer().getVideoWidth( ); 
mySurfaceViewParams. height = myUtils.getMyPlayer().getVideoHeight( ); 
} else { // 竖 屏 显 示 模 式 
//getWindow( ). clearFlags(WindowManager. LayoutParams. FLAG FULLSCREEN); 
} 
myMainLayoutParams.width = (int) myScreenWidth; 
myMainLayoutParams. height = (int) myScreenHeight; 
myMainLayout. setLayoutParams( myMainLayoutParams); 
mySurfaceView. setLayoutParams(mySurfaceViewParams); 
} 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
myUtils = new MyUtils(); 
setContentView(R. layout. activity main); 
mySurfaceView = (SurfaceView) findViewById(R. id.mySurfaceView); 
mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
myCurrentTime = (TextView) findViewById(R. id.myCurrentTime); 
myTotalTime = (TextView) findViewById(R. id.myTotalTime); 
} 
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public void onClickButtonl (View view) { // 响 应 单 击 " 选 择 视频 文件 "按钮 


Intent myIntent = new Intent(Intent. ACTION GET _CONTENT ) ; 
myIntent. setType( "video/ * "); 
myIntent. addCategory( Intent. CATEGORY OPENABLE); 
try { 

startActivityForResult( Intent. createChooser(myIntent, null), 1); 
} catch (Exception ex) { } 
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} 
public void onClickButton2(View view) { // 响 应 单 击 " 横 屏 竖 屏 切换 "按钮 
if (getResources( ) . getConfiguration( ) .orientation 
== Configuration. ORIENTRTION LANDSCAPE) { // 变 成 坚 屏 模式 
setRequestedOrientation(ActivityInfo. SCREEN ORIENTATION PORTRRIT ) ; 
} else if (getResources( ) . getConfiguration( ) . orientation 


== Configuration.ORIENTATION PORTRRIT) { // 变 成 横 屏 模式 
setRequestedOrientation(ActivityInfo. SCREEN ORIENTATION LANDSCAPE); 
} } 
(Override 


protected void onActivityResult( int requestCode, int resultCode, Intent data) { 
if (resultCode == RESULT OK) { 
myFilePath = myUtils. getPath(MainActivity. this, 
data. getData( ) ); // 获 取 视 频 文件 的 路 径 
mySurfaceView. getHolder( ) . setKeepScreenOn(true); 
mySurfaceView. getHolder().addCallback(MainActivity. this); 


上 
(Override 
public void surfaceCreated(SurfaceHolder surfaceHolder) { 
try { 
myUtils. InitMediaPlayer(new TimerTask() { 
(Override 
public void run() { // 根 据 当 前 播放 进度 设置 SeekBar 的 滑 块 位 置 
mySeekBar. setProgress(myUtils. getMediaPlayerCurrentPositionByInt( )); 
myHandler. sendEmptyMessage(2); 
} }, myFilePath, mySurfaceView. getHolder() ); 
// 显 示 视 频 时 长 
myTotalTime. setText (myUtils. getMediaPlayerDurationByString( )); 
// 根 据 视 频 时 长 设置 滑 块 最 大 值 


mySeekBar. setMax(myUtils. getMediaPlayerDurationByInt( )); 
myUtils. InitSeekBarListener(mySeekBar ); 
} catch (Exception e) { } 

} 
(Override 
public void surfaceChanged(SurfaceHolder surfaceHolder, int i, int il,int i2){ } 
(Override 
public void surfaceDestroyed( SurfaceHolder surfaceHolder) { } 


上 F 面 这 段 代 码 在 MyCode\ MySample717\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,setRequestedOrientation() 方 法 用 于 设置 屏幕 显示 方式 ， 
当 该 方法 的 参数 为 ActivityInfo. SCREEN _ORIENTATION _ LANDSCAPE 时 , 则 实现 横 屏 显示 ; 当 
其 参数 为 ActivityInfo. SCREEN_ORIENTATION _PORTRAIT 时 , 则 实现 竖 屏 显示 。getResources(). 
getConfiguration(). orientation 王 一 Configuration. ORIENTATION_LANDSCAPE 用 于 判断 当前 屏 
幕 是 否 是 横 屏 显示 ,如果 值 为 true, 则 表示 横 屏 显示 。getResources(). getConfiguration(). orientation 一 一 
Configuration. ORIENTATION_PORTRAIT 用 于 判断 当前 屏幕 是 否 是 竖 屏 显示 ,如 果 值 为 true, 则 
表示 坚 屏 显示 。myFilePath 王 myUtils. getPath(MainActivity. this ,data. getData()) 用 于 获取 在 选择 
文件 窗口 中 选择 的 视频 文件 路 径 ,myUtils 是 自 定 义工 具 类 MyUtils 的 实例 ,关于 MyUtils 类 的 详细 
内 容 请 参考 源 代 码 中 的 MyCode\MySample717\app\src\main\java\com\bin\luo\mysample\ 
MyUtils. java 文件 。 需 要 说 明 的 是 ,该 取 SD 卡 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission android:name 一 "android. permission. READ_EXTERNAL _STORAGE"/ > 权限 。 此 实例 
的 完整 项 目 在 MyCode\MySample717 文件 夹 中 。 
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191 在 选择 音乐 曲目 窗口 中 选择 音乐 文件 并 播放 


此 实例 主要 通过 使 用 MediaStore. Audio. Media. EXTERNAL_CONTENT _URI 构造 显示 所 有 
音乐 文件 的 选择 音乐 曲目 窗口 ,实现 允许 用 户 选 择 音乐 文件 并 播放 。 当 实例 运行 之 后 , 单 击 “ 选 择 并 
播放 音乐 文件 ”按钮 , 则 将 显示 “选择 音乐 曲 日 ”窗口 ,如 图 191. 1 的 左 图 所 示 。 在 音乐 曲目 列表 中 任 
意 选 择 一 个 音乐 文件 ,如 “alarm01. wav”, 然 后 单 击 “ 确 定 ” 按 钮 , 则 将 播放 此 音乐 文件 ,如 图 191. 1 的 
右 图 所 示 。 


三 六 = A 区 
my? 日 6 oO 下 午 541| 四 加 四 
' “ 1 ~ -4 . 


选择 音乐 曲目 ; | MySample 


选择 并 播放 音乐 文件 


storage/sdcard0/alarm01.wav 


图 191. 1 
主要 代码 如 下 : 


public void onClickmyBtnl(View v) { // 响 应 单 击 "选择 并 播放 音乐 文件 "按钮 
Intent myIntent = new Intent(Intent.ACTION PICK，nul1); 
myIntent. setDataAndType( MediaStore. Rudio. Media. EXTERNAL CONTENT URI，nul1); 
startActivityForResult(myIntent, 1); 


} 
(Override 
protected void onActivityResult( int myRequestCode ， 
int resultCode, Intent myIntent) { // 播 放 选 择 的 音乐 文件 
super. onActivityResult(myRequestCode, resultCode, myIntent); 
try { 
switch (myRequestCode) { 


case 1: 
if (resultCode == Activity.RESULT OK) { 
Uri myUri = myIntent. getData( ); 
String[ ] myPathColumns = {MediaStore. Images. Media.DATA}); 
Cursor myCursor = 
getContentResolver().query(myUri, myPathColumns, null, null, null]l); 
myCursor. moveToFirst( ); 
int myIndex = myCursor.getColumnIndex(myPathColumns[0]); 
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String myFilePath = myCursor.getString(myIndex); 
MediaPlayer myMediaPlayer = new MediaPlayer( ); 
myMediaPlayer. setDataSource(myFilePath); 
myMediaPlayer. prepare( ); 
myMediaPlayer. start( ); 
myTextView. setText(myFilePath); 

} 

break; 

} 
} catch (Exception e){ e. printStackTrace(); } 
} 


// 设 置 音乐 文件 路 径 


上 面 这 段 代 码 在 MyCode\ MySample380 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myIntent 一 new Intent(Intent. ACTION_PICK ,null) 用 
于 创建 可 进行 选择 的 Intent。myIntent. setDataAndType(MediaStore. Audio. Media. EXTERNAL _ 
CONTENT_URI, nulD) 表 示 此 Intent 的 数据 类 型 为 音乐 文件 。onActivityResult() 用 于 处 理 用 户 在 
“选择 音乐 曲目 ”窗口 中 单 击 “ 确 定 ” 或 “取消 ”按钮 之 后 返回 的 信息 ,如 音乐 文件 全 路 径 信 息 等 。 此 外 ， 
读 取 在 SD 卡 上 的 音乐 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 二 
"android. permission. READ_EXTERNAL STORAGE"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample380 文件 夹 中 。 


192 ”在 RecyclerView 中 加 载 音 乐 文件 并 播放 


此 实例 主要 通过 自 定义 数据 适 配 右 ,实现 在 RecyclerView 的 列表 中 显示 当前 手机 的 所 有 音乐 文 
件 , 选 择 任 一 列表 项 (音乐 文件 ) , 则 播放 此 音乐 文件 。 当 实例 运行 之 后 ,将 在 RecyclerView 的 列表 中 
显示 当前 手机 的 所 有 音乐 文件 ,选择 任 一 列表 项 (音乐 文件 ), 则 立即 播放 音乐 ,在 底部 的 SeekBar 也 
将 同步 显示 播放 进度 , 单 击 右 侧 的 暂停 或 播放 按钮 则 可 暂停 或 播放 音乐 ,效果 分 别 如 图 192. 1 的 左 图 
和 右 图 所 示 。 
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主要 代码 如 下 : 
public class MainActivity extends Activity { 
Handler myHandler = new Handler() { 
(Override 
public void handleMessage( Message msg) { // 发 送 消息 更 新 播放 时 间 


if (msg.what == 1) { 
String myCurrentSeconds = myMediaPlayer.getCurrentPosition() / 1000 /60 + ":"; 
if (myMediaPlayer. getCurrentPosition() / 1000 % 60<10) { 
myCurrentSeconds += "0" + myMediaPlayer. getCurrentPosition() / 1000 % 60; 
} else { myCurrentSeconds += myMediaPlayer. getCurrentPosition() / 1000 % 60; } 
myCurrentTime. setText (myCurrentSeconds); 
be 
MediaPlayer myMediaPlayer; 
Timer myTimer; 
ArrayList <MyMusicInfo> myData = new ArrayList <>(); 
private TextView myCurrentTime; 
boolean isPlaying = false; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
myMediaPlayer = new MediaPlayer(); 
myTimer = new Timer(); 
setContentView(R. layout.activity main); 
RecyclerView myRecyclerView = (RecyclerView) findViewById(R. id.myRecyclerView); 
final TextView myTotalTime = (TextView) findViewById(R. id.myTotalTime); 
myCurrentTime = (TextView) findViewById(R. id.myCurrentTime); 
final SeekBar mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
myRecyclerView. setLayoutManager(new LinearLayoutManager (this) ); 
final MyAdapter myAdapter = new MyAdapter(this); 
new MyUtils(this).getMusicInfo(myData); 
myAdapter. setMyData( myData); 
myRecyclerView. setAdapter(myAdapter); 
final ImageView myPlayPause = (ImageView) findViewById(R. id.myPlayPause); 
myAdapter. setMyListener(new MyAdapter. MyOnItemClickListener( ){ 
(Override 
public void onItemClick(View v, int position){ 
try { 
if (myMediaPlayer != null || myTimer != null){ 
myTimer. cancel( ); 
myTimer = null; 
myMediaPlayer. stop( ) ; 
myMediaPlayer = null; 
myPlayPause. setImageResource(R. mipmap. myicon play); 
} 
myMediaPlayer = new MediaPlayer( ); 
myMediaPlayer. setDataSource(myData. get(position).myMusicPath); 
myMediaPlayer. prepare( ); 
myMediaPlayer. start( ); 
isPlaying = true; 
mySeekBar. setMax( myMediaPlayer. getDuration( ) ) ; 
mySeekBar. setOnSeekBarChangeListener(new SeekBar. OnSeekBarChangeListener( ){ 
(Override 
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public void onProgressChanged( SeekBar seekBar, int i, boolean b) { } 
(@ Override 
public void onStartTrackingTouch( SeekBar seekBar) { } 
(@Override 
public void onStopTrackingTouch( SeekBar seekBar) { 
myMediaPlayer. seekTol( seekBar. getProgress( ) ); 
} }); 
String myDurations = myMediaPlayer.getDuration() / 1000/ 60 +":"; 
if (myMediaPlayer.getDuration() / 1000 % 60<10) { 
myDurations += "0" + myMediaPlayer.getDuration() / 1000 % 60; 
} else { myDurations += myMediaPlayer.getDuration() / 1000 % 60; } 
myTotalTime. setText(myDurations); 
myPlayPause. setImageResource(R. mipmap. myicon pause); 
myTimer = new Timer(); 
myTimer. Schedule(new TimerTask() { 
(Override 
public void run() { 
mySeekBar. setProgress(myMediaPlayer. getCurrentPosition( ) ) ; 
myHandler. sendEmptyMessage(1); 
} }, 0, 10); 
myAdapter. notifyDataSetChanged( ); 
} catch (Exception e) { } 
} }); 
myPlayPause. setOnClickListener(new View. OnClickListener() { 
(Override 
public void onClick(View view) { 
if (isPlaying) { // 如 果 正 在 播放 , 则 暂停 
isPlaying = false; 
myPlayPause. setImageResource(R.mipmap. myicon play); 
myMediaPlayer. pause( ); 
} else { // 如 果 已 经 暂停 , 则 播放 
isPlaying = true; 
myPlayPause. setImageResource(R. mipmap. myicon pause); 
myMediaPlayer. start( ); 
ep 


上 和 面 这 段 代 人 码 在 MyCode\ MySample714\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myRecyclerView. setAdapter(myAdapter) 表 示 使 用 自 定 
义 适 配器 MyAdapter 的 实例 myAdapter 作为 RecyclerView 的 适 配 需 ,MyAdapter 上 自 定 义 适 配器 的 
主要 代码 如 下 : 


public class MyAdapter extends RecyclerView. Adapter < MyAdapter. MyViewHolder >{ 
Context myContext; 
ArrayList <MyMusicInfo> myData = new ArrayList <MyMusicInfo>(); 
MyOnItemClickListener myListener = null; 
public MyAdapter(Context context){ myContext = context; } 
public void setMyData(ArrayList < MyMusicInfo> data){ 
myData = data; 
notifyDataSetChanged( ) ; 
(Override 
public MYViewHolder onCreateViewHolder(ViewGroup parent, int viewType){ 
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return new MyViewHolder( 
LayoutInflater. from(myContext). inflate(R. layout. myitem, parent, false)); 
} 
public void setMyListener(MyOnItemClickListener listener)!{ 
myListener = listener; 
} 
人 Override 
public void onBindViewHolder(MyViewHolder holder, int position) { 
holder. myArtist. setText(myData. get(position).myArtist); 
holder. myDuration. setText(myData. get(position).myDuration); 
holder. myName. setText (myData. get(position). myName); 
if (myData. get(position). getSelected()) { 
holder. myItemView. setBackgroundColor (Color. LTGRAY); 
} else { holder. myItemView. setBackgroundColor(Color. WHITE); } 
} 
(Override 
public int getItemCount( ) { return myData. size(); } 
class MyViewHolder extends RecyclerView. ViewHolder { 
private final TextView myArtist; 
private final TextView myDuration; 
private final TextView myName; 
View myItemView; 
public MYViewHolder(View itemView) { 


super (itemView); 

myArtist = (TextView) itemView.findViewById(R. id.myArtist); // 艺 术 家 
myDuration = (TextView) itemView.findViewById(R. id.myDuration); // 音 乐 时 长 
myName = (TextView) itemView.findViewById(R. id.myName); // 音 乐 名 称 


myItemView = itemView; 
itemView. setOnClickListener(new View. OnClickListener() { 
(QOverride 
public void onClick(View view) { // 响 应 单 击 列表 项 事件 ( 即 选择 文件 
并 播放 ) 
for (int i = 0; i< myData. size(); i++) {myData.get(i).setSelected(false); } 
myData. get (getAdapterPosition( )). setSelected(true); 
myListener. onItemClick(view, getAdapterPosition( ) ); 


Pl 
interface MyOnItemClickListener{ void onItemClick(View v, int position);} 


上 面 这 段 代 码 在 MyCode\ MySample714\app\src\main\java\com\bin\luo\ mysample\ 
MyAdapter. java 文件 中 。 在 这 段 代 码 中 ,new MyViewHolder(LayoutInflater. from (myContext ). 
inflate(R. layout. myitem，parent，false) ) 用 于 加 载 myitem 布局 以 创建 MyAdapter 的 列表 项 , 即 每 
个 列表 项 的 布局 。 关 于 myitem 布局 的 内 容 请 参考 源 代 码 中 的 MyCode\MySample714\app\src\main 
\res\layout\myitem. xml 文件 。 在 MyAdapter. java 文件 中 ,ArrayList< MyMusicInfo > myData = 
new ArrayList < MyMusicInfo >() 用 于 创建 一 个 数组 ,以 保存 音乐 文件 的 myName、myArtist、 
myDuration .myMusicPath \isSelected 等 属性 ,从 而 决定 是 否 播 放 或 暂停 。MyMusiclInfo 是 自 定义 
类 ,关于 MyMusicInfo 类 的 详细 内 容 请 参考 源 代 码 中 的 MyCode\MySample714\app\src\main\java\ 
com\bin\luo\mysample\ MyMusicInfo. java 文件 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 
gradle 中 引入 compile 'com. android. support: design:25. 0.1' 依 赖 项 。 此 外 , 读 取 在 SD 卡 上 的 音乐 
文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 一 " android. 
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permission. READ EXTERNAL _ STORAGE "> 权限 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample714 文件 夹 中 。 


193 ”依次 播放 在 RecyclerView 中 的 音乐 文件 


此 实例 主要 通过 在 MediaPlayer 的 setOnCompletionListener() 方 法 中 设置 播放 曲目 ,实现 按照 
顺序 播放 RecyclerView 加 载 的 音乐 文件 。 当 实例 运行 后 ,将 在 RecyclerView 的 列表 中 显示 当前 手机 
的 所 有 音乐 文件 ,选择 任 一 列表 项 (音乐 文件 ), 则 立即 播放 音乐 ,在 底部 的 SeekBar 也 将 同步 显示 播 
放 进 度 , 单 击 右 侧 的 暂停 或 播放 按钮 则 可 暂停 或 播放 音乐 ; 如 果 当 前 曲目 播放 完毕 , 则 自动 播放 列表 
中 的 下 一 曲目 ,效果 分 别 如 图 193. 1 的 左 图 和 右 图 所 示 。 


MySample 
05. 鞍 程 礼 理 之 线程 1- 操 人 区 尿 壹 ) ;学 正 ， , 进 栏 当 理 之 线 样 1-3 
2013] 标清 .mp3 2013] 标清 .mp3 
<UNKNOWN> <Unknown> 


06. 进 程 管理 之 线程 2- 操 作 系 统 原理 [天 津 大 学 , 李 定 ， 06. 进 程 管理 之 线程 2- 操 作 系 统 原 理 [ 天 津 大 学 , 李 畦 , 
2013] 标清 .mp3 2013] 标清 .mp3 


47:28 | <unknown> 47:28 


mymusic1.mp3 mymusic1.mp3 

Don Omar S00 | Don Omar 5:06 

mymusic2.mp3 

Need For Speed Need For Speed 3:08 

mymusic3.mp3 mymusic3.mp3 

Need For Speed Need For Speed 4:46 
6:17 
2:05 


<UNKNOWN> 


0 
4 
mymusic5.mp3 mymusic5.mp3 
Need For Speed 0 Need For Speed 
mysound.mp3 mysound.mp3 
站 长 素材 (sc.chinaz.com) 0:0 站 长 素材 (sc.chinaz.com) 0:07 


Don OmarTego Calder6n - Bandoleros [mqms2].mp3 | Don OmarTego Calderon - Bandoleros [mqms2].mp3 
Don Omar 5:06 | Don Omar 5:06 


EA Games Soundtrack - Crossover [mqms2].mp3 EA Games Soundtrack - Crossover [mqms2].mp3 


3:08 
4:46 
mymusic4.mp3 mymusic4.mp3 
Need For Speed 6:17 || Need For Speed 
2:05 
7 
0 


图 193.1 
主要 代码 如 下 : 


public class MainActivity extends Activity Implements View. OnClickListener, 
SeekBar. OnSeekBarChangeListener { 
Handler myHandler = new Handler() { 
(QOverride 
public void handleMessage(Message msg) { 
myCurrentTime. setText(myUtils. getMediaPlayerCurrentPositionByString( ) ) ; 
} }; 
private TextView myTotalTime, myCurrentTime; 
private MyUtils myUtils; 
private ImageView myPlayPause; 
private MyAdapter myAdapter; 
private SeekBar mySeekBar; 
int myIndex; 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
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myUtils = new MyUtils(this); 
setContentView(R. layout. activity main); 
mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
myTotalTime = (TextView) findViewById(R. id.myTotalTime); 
myPlayPause = (ImageView) findViewById(R. id.myPlayPause); 
myCurrentTime = (TextView) findViewById(R. id.myCurrentTime); 
myPlayPause. setOnClickListener(this); 
mySeekBar. setOnSeekBarChangeListener (this); 
RecyclerView myRecyclerView = (RecyclerView) findViewBylId(R. id.myRecyclerView); 
myRecyclerView. setLayoutManager (new LinearLayoutManager(this)); 
myadapter = new MyAdapter(this); 
myAdapter. setMyData(myUtils. getMyData( ) ) ; 
myRecyclerView. setAdapter(myAdapter); 
myAdapter. setMyListener(new MyAdapter. MyOnItemClickListener() { 
(@ Override 
public void onItemClick(View v, int position) { 
selectMusicItem( position); // 选 择 曲 目 列表 选项 
Te 
public void selectMusicItem( int index) { 
ArrayList < MYMusicInfo > myData = myAdapter. getMyData( ) ; 
for (int i = 0; i< myData. size(); i++) { myData.get(i). setSelected(false); } 
myData. get (index). setSelected(true); 
myAdapter. setMyDatal( myData); 


try { 
myUtils. InitMyPlayer(myData. get(index) .myPath，new TimerTask() { 
(Override 
public void run() { // 根 据 播放 进度 设置 SeekBar 滑 块 位 置 
mySeekBar. setProgress(myUtils. getMediaPlayerCurrentPositionByInt( )); 
myHandler. sendEmptyMessage(1); 
1 3) 
mySeekBar. setMax(myUtils. getMyPlayer().getDuration( )); // 根 据 时 长 设置 最 大 值 


myTotalTime. setText (myUtils. getMediaPlayerDurationByString( )); 
myPlayPause. setImageResource(R. mipmap. myicon pause); 
myIndex = index; 
myUtils. getMyPlayer(). setOnCompletionListener!( 
new MediaPlayer. OnCompletionListener() { // 当 播放 完成 之 后 继续 播放 下 一 曲目 
(Override 
public void onCompletion(MediaPlayer mediaPlayer) { 
mediaPlayer. stop( ); 
selectMusicItem(myIndex + 1 < myUtils.getMyData().size() ? myIndex + 1 : 0); 
} }); 
}catch (Exception e) { } 
myAdapter. notifyDataSetChanged( ); 
} 
(Override 
public void onClick(View view) { 
if (view. getId() == R. id.myPlayPause) { 
if (myUtils.getMyPlayer().isPlaying()) { // 在 暂停 音乐 时 设置 播放 图 标 
myPlayPause. setImageResource(R. mipmap. myicon play); 
myUtils. getMyPlayer(). pause( ); 
} else { // 在 播放 音乐 时 设置 暂停 图 标 
myPlayPause. setImageResource(R. mipmap. myicon pause); 
myUtils. getMyPlayer(). start(); 
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Override 
public void onProgressChanged( SeekBar seekBar, int i, boolean b) { } 
(QOverride 
public void onStartTrackingTouch( SeekBar seekBar) { } 
(Override 
public void onStopTrackingTouch( SeekBar seekBar) { 
myUtils. getMyPlayer( ) . seekTo( seekBar. getProgress( ) ) ; 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample716\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myRecyclerView. setAdapter(myAdapter) 表 示 使 用 自 定 
义 适 配器 MyAdapter 的 实例 myAdapter 作为 RecyclerView 的 适 配 寓 ,关于 MyAdapter 自 定 义 适 配 
器 的 详细 内 容 请 参考 源 代 码 中 的 MyCode\MySample716\app\src\ main\java\com\bin\luo\ 
mysample\MyAdapter. java 文件 。 需 要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 
'com. android. support: design: 25. 0. 1 ' 依赖 项 。 此 外 , 读 取 在 SD 卡 上 的 音乐 文件 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 王 "android. permission. READ 


EXTERNAL _ STORAGE"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample716 文件 夹 中 ， 


194 在 ListView 上 加 载 手 机 外 存 的 音乐 文件 


此 实例 主要 是 实现 在 ListView 上 加 载 手机 外 部 存储 介质 上 的 所 有 音乐 文件 名 称 和 大 小 信息 。 
当 实 例 运行 后 ,将 在 ListView 控件 上 显示 手机 外 部 存储 上 的 所 有 音乐 文件 , 单 击 在 ListView 控件 上 
的 任意 音乐 文件 名 称 ( 列 表 项 ) ,将 播放 此 音乐 文件 ,效果 分 别 如 图 194. 1 的 左 图 和 右 图 所 示 。 


全 山口 里 四 0 玖 下 WOQ 


MySample MySample 


mymusic3.mp3 7823991 mymusic3.mp3 7823991 
mymusic1.mp3 6675021 mymusic1.mp3 6675021 
mymusic2.mp3 4287829 mymusic2.mp3 4287829 


mymusic2.mp3 4287829 mymusic2.mp3 4287829 


图 194.1 
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public class MainActivity extends AppCompatActivity { 
private ListView myListView; 
(Override 
protected void onCreate( @Nullable Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main) ; 
myListView = (ListView) findViewById(R. id. myListView) ; 
GetAudioFiles( ); 
} 
ArrayList myFiles = new ArrayList(); 
ArrayList mySizes = new ArrayList(); 
ArrayList < String> myPathData = new ArrayList(); 
ArrayList < HashMap < String, Object >> myItems = new ArrayList <>(); 
private void GetAudioFiles() { 
myFiles. clear( ); 
mySizes. clear( ); 
// 通 过 ContentResolver 查询 所 有 外 部 存储 音乐 文件 信息 
Cursor myCursor = getContentResolver().query( 
MediaStore. Audio. Media. EXTERNAL CONTENT URI, null, null, null, null); 
while (myCursor.moveToNext()) { 
String myName = myCursor.getString(myCursor.getColumnIndex( 
MediaStore. Images. Media.DISPLAY NAME));  // 显 示 文 件 名 称 
String mySize = myCursor.getString(myCursor.getColumnIndex( 


MediaStore. Images. Media. SIZE) ); // 显 示 文 件 大 小 
byte[ ] data = myCursor.getBlob(myCursor.getColumnIndex( 
MediaStore. Images. Media. DATA) ) ; // 获 取 音 乐 文件 的 位 置 数据 

myFiles.add(myName); 

mySizes.add(mySize); 

String myPath = new String(data, 0, 

data. length - 1); // 将 位 置 数 据 转换 成 字符 串 形式 的 路 径 

myPathData. add( myPath); 


} 

for (int i = 0; i< myFiles. size(); i++) { 

HashMap < String, Object > myListItem = new HashMap <>(); 
myListItem.put("name", myFiles.get(i)); 

myListItem. put(" size", mySizes.get(i)); 

myItems. add(myListItem); 

} 

SimpleAdapter myAdapter = new SimpleAdapter(MainActivity.this, 
myItems, R. layout.myitem, new String[ ]{" name", "size"}, 
new int[ ]{R. id.myName, R. id.mySize} ); 

myListView. setAdapter( myAdapter ); 

myListView. setOnItemClickListener(new AdapterView.OnItemClickListener() { 


(@ Override 
public void onItemClick(AdapterView <?> parent, View view, 
int position, long id) { // 单 击 音乐 文件 名 称 (列表 项 ) 播 放 音 乐 
try{ 


MediaPlayer myMediaPlayer = new MediaPlayer( ); 
myMediaPlayer. setDataSource(myPathData. get(position)); 
myMediaPlayer. prepare( ); 
myMediaPlayer. start( ); 
} catch (Exception e) { e.printStackTrace( ); } 
ry 
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上 F 面 这 段 代 码 在 MyCode\ MySample669\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 上 面 这 段 代 码 中 ,myAdapter 一 new SimpleAdapter(MainActivity. 
this, myltems, R. layout. myitem, new String| |{"name", "size"},new int| | {R. id. myName, R. 
id. mySize}) ) 表 示 根 据 myitem 布局 的 样式 创建 ListView 的 列表 项 ,关于 myitem 布局 的 详细 内 容 请 
参考 源 代 码 中 的 MyCode\MySample669\app\src\main\res\ layout\myitem. xml 文件 中 。 此 外 , 读 
取 SD 卡 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android:name 一 "android. 
permission. READ_EXTERNAL _ STORAGE"/ > 权限。 需要 说 明 的 是 ,使 用 此 实例 的 相关 类 需要 在 
gradle 中 引入 compile 'com. android. support: design: 25. 0. 1 ' 人 依赖 项 。 此 实例 的 完整 项 目 在 
MyCode\MySample669 文件 夹 中 。 


195 ”使 用 SoundPool 播放 较 短 的 再 音 片 段 


此 实例 主要 通过 使 用 SoundPool, 实 现 从 程序 的 资源 中 加 载 并 播放 较 短 的 声音 片段 。 当 实例 运行 
之 后 , 单 击 “ 播 放 第 一 种 声音 两 次 ”按钮 ,将 播放 声音 片段 两 次 ; 单 击 “ 播 放 第 二 种 声音 一 次 ”按钮 ,将 仅 
播放 一 次 声音 片段 ,效果 分 别 如 图 195. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


播放 第 一 种 声音 两 次 播放 第 一 种 声音 两 次 
播放 第 二 种 声音 一 次 播放 第 二 种 声音 一 次 


正在 描 放 重复 的 声 昔 ! 


图 195. 1 
主要 代码 如 下 : 


public void onClickmyBtnl (View v) { // 响 应 单 击 "播放 第 一 种 声音 两 次 "按钮 
Toast. makeText (myContext, "正在 播放 重复 的 声音 !", Toast. LENGTH_SHORT). show() ; 
playSound(1, 2); 

} 

public void onClickmyBtn2(View v) { // 响 应 单 击 "播放 第 二 种 声音 一 次 "按钮 
Toast. makeText (myContext, "正在 播放 无 重复 声音 !",Toast. LENGTH SHORT) . show( ) ; 
playSound(2, 1); 

} 

public void InitSound() { // 初 始 化 声音 
mySoundPool = new SoundPool(5, AudioManager.STREAM MUSIC, 0); 
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myHashMap = new HashMap < Integer, Integer >(); 
myHashMap. put(1，mySoundPoo1l. load(this，R. raw. alarm01，1) ); 
myHashMap. put(2, mySoundPool]. load(this, R.raw.alarm02, 1)); 
mySoundPool. setOnLoadCompleteListener(new SoundPool. OnLoadCompleteListener( ){ 
(@ Override 
public void onLoadCompletel( SoundPool] sound, int sampleId, int status){ 
isLoaded = true; 
Toast. makeText(myContext," 音 效 加 载 完成 !" ,Toast. LENGTH SHORT) . show( ); 
} }); } 
public void playSound( int sound, int number) { // 播 放声 音 
AudioManager myAudioManager = 
(AudioManager) this. getSystemService(Context. AUDIO SERVICE); 
float myMaxVolume = 
myAudioManager. getStreamMaxVolume( AudioManager. STREAM MUSIC); 
float myCurVolume = myAudioManager.getStreamVolume(AudioManager. STREAM MUSIC); 
float myRatio = myCurVolume/ myMaxVolume; 
mySoundPool. play( myHashMap. get( sound), 


myRatio, // 左 声 道 音量 

myRatio, // 右 声 道 音量 

用 // 优 先 级 

number, // 循 环 播放 次 数 

1); // 回 放 速 度 , 该 值 在 0.5 一 2.0 之 间 1 为 正常 速度 


上 面 这 段 代 码 在 MyCode\ MySample375 \app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , mySoundPool = 二 new SoundPool(5，AudioManager. 
STREAM_MUSIC, 0) 用 于 创建 一 个 SoundPool 对 象 ,SoundPool() 构 造 函 数 的 语法 声明 如 下 : 


public SoundPool( int maxStream, int streamType, int srcQuality) 


其 中 ,参数 int maxStream 表示 同时 播放 流 的 最 大 数量 。 参 数 int streamType 表示 流 的 类 型 ,一 
般 为 STREAM_MUSIC( 具 体 在 AudioManager 类 中 列 出 )。 参 数 int srcQuality 表示 采样 率 转化 质 
量 ,使 用 0 作为 默认 值 。 

mySoundPool. load(this，R. raw. alarm01, 1) 表 示 从 资源 R. raw. alarm01 加 载 音 效 。 如 果 使 用 
load() 方 法 从 文件 中 加 载 音效 , 则 其 语法 为 load(String path，int priority)。 

mySoundPool. play(myHashMap. get(sound),myRatio, myRatio,1,number,1) 用 于 播放 音效 ， 
play() 方 法 的 语法 声明 如 下 : 


play( int soundID, float leftVolume, float rightVolume, 
int priority, int loop, float rate) 


其 中 ,参数 int soundID 用 于 指定 音效 ID, 即 在 加 载 音效 资源 时 设置 的 ID 值 。 参 数 float 
leftVolume 和 参数 float rightVolume 表示 左右 声 道 音量 。 参 数 int priority 表示 优先 级 。 参 数 int 
loop 表示 循环 次 数 。 参 数 float rate 表示 速率 ,速率 最 低 0.5, 最 高 为 2,1 代表 正常 速度 。 

此 实例 的 完整 项 目 在 MyCode\MySample375 文件 夹 中 。 


196 ”使 用 AudioManasger 增 大 或 减 小 音量 


此 实例 主要 通过 使 用 AudioManager 的 adjustStreamVolume() 方 法 ,实现 直接 调 出 系统 音量 控 
制 条 增 大 或 减 小 音量 。 在 实例 运行 之 前 , 先 使 用 任 一 音乐 播放 需 播 放歌 曲 ; 当 实 例 运 行 之 后 , 单 击 “ 减 
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小 音量 ”按钮 , 则 在 顶部 显示 系统 默认 音量 控制 条 并 减 小 音量 ; 单 击 “ 增 大 音量 ”按钮 ,也 将 在 项 部 显示 
系统 默认 音量 控制 条 并 增 大 音量 ,效果 分 别 如 图 196. 1 的 左 图 和 右 图 所 示 。 


减 小 音量 减 小 音量 


图 196.1 
主要 代码 如 下 : 
public void onClickButtonl (View v) { // 响 应 单 击 " 减 小 音量 "按钮 
AudioManager myAudioManager = 


(AudioManager)getSystemService(Context. AUDIO SERVICE); 
myAudioManager.adjustStreamVolume( AudioManager. STREAM MUSIC., 
AudioManager. ADJUST LOWER, AudioManager. FX FOCUS NAVIGATION UP); 


} 
public void onClickButton2(View v) { // 响 应 单 击 " 增 大 音量 "按钮 


AudioManager myAudioManager = 
(AudioManager)getSystemService(Context. AUDIO SERVICE); 
myAudioManager.adjustStreamVolume( AudioManager. STREAM MUSIC., 
AudioManager. ADJUST RAISE, AudioManager. FX FOCUS NAVIGATION UP); 


上 面 这 段 代 码 在 MyCode\ MySample064\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,adjustStreamVolume(int streamType, int direction, int 
flags) 方 法 用 于 以 步 长 方式 调节 手机 音量 大 小 。 其 中 ,参数 bn 表示 声音 类 型 ,可 取 值 为 
STREAM_VOICE_CALL (通话 )、STREAM _SYSTEM (系统 声音 )、STREAM _RING (铃声 )、 
STREAM_MUSIC( 音 乐 )、STREAM_ALARM( 曾 铃声 ); 参数 direction 用 于 设置 调整 音量 的 方向 ， 
可 取 值 为 ADJUST_LOWER( 减 小 )\ADJUST _ RAISE( 增 大 )、 ADJUST_SAME 等 ; 参数 flags 表示 
可 选 标 志 位 , 当 此 值 为 FX_FOCUS_NAVIGATION_UP 时 ,将 在 增 大 或 减 小 音量 时 显示 系统 默认 音 
量 控 制 条 , 当 此 值 为 FX_FOCUS_NAVIGATION_DOWN 时 ,将 在 增 大 或 减 小 音量 时 隐藏 系统 默认 

音量 控制 条 ,此 参数 还 支持 FX_KEY_CLICK 、FX_KEYPRESS_STANDARD、FX_FOCUS _ 
NAVIGATION RIGHT FX FOCUS _ NAVIGATION _LEFT、FX_ KEYPRESS SPACEBAR.\.FX _ 
KEYPRESS DELETE、 FX_KEYPRESS_ RETURN 等 。 此 外 ,adjustVolume() 方 法 也 能 实现 与 
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adjustStreamVolume() 方 法 类 似 的 调节 音量 的 功能 。 此 实例 的 完整 项 目 在 MyCode\ MySample064 
文件 夹 中 ，。 


197 使 用 AudioManasger 播放 系统 预 置 的 再 将 


此 实例 主要 通过 使 用 AudioManager 的 playSoundEffect() 方 法 ,实现 直接 播放 系统 预 置 的 删除 、 
返回 等 音效 。 当 实例 运行 后 , 单 击 “播放 删除 音效 ?按钮 , 则 将 播放 删除 音效 ; 单 击 “ 播 放 人 返回 音效 ” 按 
钮 , 则 将 播放 返回 音效 ,效果 分 别 如 图 197. 1 的 左 图 和 右 图 所 示 。 


中 回 和 
t | 


MySample MySample 


播放 删除 音效 播放 返回 音效 播放 删除 音效 播放 返回 音效 


正在 播放 删除 音效 .… 


图 197.1 
主要 代码 如 下 : 


public void onClickButtonl(View v) { // 响 应 单 击 "播放 删除 音效 "按钮 
AudioManager myAudioManager = 
(AudioManager)getSystemService( Context. AUDIO SERVICE); 
myAudioManager. playSoundEffect(AudioManager. FX KEYPRESS DELETE,1000); 


Toast. makeText(this," 正 在 播放 删除 音效 ...... ",Toast. LENGTH SHORT) . show( ) ; 

} 

public void onClickButton2(View v) { // 响 应 单 击 "播放 返回 音效 "按钮 
AudioManager myAudioManager = 


(AudioManager)getSystemService(Context. AUDIO SERVICE); 
myAudioManager. playSoundEffect( AudioManager. FX KEYPRESS RETURN,1000); 
Toast. makeText(this, "正在 播放 返回 音效 .…...", Toast. LENGTH_SHORT). show( ); 

} 


上 面 这 上段 代码 在 MyCode\ MySample065\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,playSoundEffect(int effectType,，float volume) 方 法 用 于 
播放 系统 预 置 的 音效 。 其 中 ,参数 int effectType 表示 音效 类 型 ,可 选 值 有 AudioManager. FX_KEY_ 
CLICK AudioManager. FX KEYPRESS_ RETURN., AudioManager. FX_FOCUS NAVIGATION_ 
DOWN., AudioManager. FX FOCUS NAVIGATION __ LEFT AudioManager. FX FOCUS _ 
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NAVIGATION _ RIGHT、AudioManager. FX KEYPRESS STANDARD, AudioManager. FX _ 
KEYPRESS SPACEBAR., AudioManager. FX KEYPRESS DELETE, AudioManager. FX FOCUS 
_NAVIGATION UP 和 AudioManager. FX KEYPRESS_INVALID 等 。 此 实例 的 完整 项 目 在 
MyCode\MySample065 文件 夹 中 。 


198 ”使 用 AudioManasger 获取 和 设置 铃声 模式 


此 实例 主要 通过 使 用 AudioManager 的 getRingerMode() 方 法 和 setRingerMode() 方 法 ,获取 和 
设置 当前 手机 的 铃声 模式 。 当 实例 运行 之 后 , 单 击 “ 设 置 当 前 铃声 模式 为 振动 模式 ”按钮 ,将 设置 当前 
手机 铃声 模式 为 振动 模式 ,如 图 198. 1 的 左 图 所 示 ; 单 击 “获取 当前 铃声 模式 ”按钮 , 则 在 弹出 的 
Toast 中 显示 “当前 手机 铃声 模式 是 : 振动 模式 ”, 如 图 198. 1 的 右 图 所 示 。 单 击 其 他 按钮 则 会 实现 按 
钮 标题 所 示 的 功能 。 需 要 说 明 的 是 ,此 实例 在 模拟 器 中 测试 可 能 不 会 成 功 ,建议 直接 在 Android 手机 


中 测试 。 
MySample MySample 
获取 当前 铃声 模式 获取 当前 铃声 模式 
设置 当前 铃声 模式 为 静音 模式 设置 当前 铃声 模式 为 静音 模式 
设置 当前 铃声 模式 为 振动 模式 设置 当前 铃声 模式 为 振动 模式 
设置 当前 铃声 模式 为 普通 模式 设置 当前 铃声 模式 为 普通 模式 
已 经 设置 当前 铃声 模式 为 振动 模式 当前 手机 铃声 模式 是 : 振动 模式 
图 198. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
public AudioManager myAudioManager; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity_main); 
myAudioManager = (AudioManager) getSystemService(Context. AUDIO SERVICE); 
} 
public void onClickmyBtnGetRingerMode(View v) { // 响 应 单 击 "获取 当前 铃声 模式 "按钮 
int myMode = myAudioManager. getRingerMode( ); 
String myText = "当前 手机 铃声 模式 是 :"; 
switch (myMode) { 
case AudioManager. RINGER MODE NORMAL: 
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myText += "普通 模式 "; 
break; 
case AudioManager. RINGER MODE SILENT: 
myText += "静音 模式 "; 
break; 
case AudioManager. RINGER MODE VIBRATE: 
myText += "振动 模式 "; 
break; 
default: 
myText += "未 知 模式 "; 
break; 
} 
Toast. makeText (getApplicationContext(), myText, Toast.LENGTH SHORT). show(); 
} 
// 响 应 单 击 " 设 置 当 前 铃声 模式 为 静音 模式 "按钮 
public void onClickmyBtnSetRingerModeSilent(View v) { 
myAudioManager. setRingerModel( AudioManager. RINGER MODE SILENT ) ; 
Toast. makeText (getApplicationContext( ) ， 
"已 经 设置 当前 铃声 模式 为 静音 模式 "，Toast. LENGTH_SHORT). show( ); 


} 
// 响 应 单 击 "设置 当前 铃声 模式 为 振动 模式 "按钮 
public void onClickmyBtnSetRingerModeVibrate(View v) { 
myAudioManager. setRingerMode( AudioManager. RINGER MODE VIBRATE); 
Toast. makeText (getApplicationContext( ) ， 
"已 经 设置 当前 铃声 模式 为 振动 模式 "，Toast. LENGTH_SHORT). show( ) ; 


} 
// 响 应 单 击 "设置 当前 铃声 模式 为 普通 模式 "按钮 
public void onClickmyBtnSetRingerModeNormal (View v) { 
myAudioManager. setRingerMode( AudioManager. RINGER MODE NORMAL); 
Toast. makeText (getApplicationContext( ) ， 
"已 经 设置 当前 铃声 模式 为 普通 模式 "，Toast. LENGTH_SHORT). show( ); 
} } 


上 面 这 上段 代码 在 MyCode\ MySample066\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,getRingerMode() 方 法 用 于 获取 当前 手机 的 铃声 模式 ,可 
以 通过 该 方法 的 返回 值 来 判断 当前 手机 的 铃声 模式 ,如 果 返 回 值 是 RINGER_MODE_NORMAL, 则 
手机 铃声 是 普通 模式 ; 如 果 返 回 值 是 RINGER MODE_SILENT, 则 手机 铃声 是 静音 模式 、 如 果 返 回 
值 是 RINGER_MODE_VIBRATE, 则 手机 铃声 是 振动 模式 。setRingerMode(int Mode) 方 法 用 于 设 
置 手 机 铃声 模式 ,该 方法 的 参数 Mode 表示 铃声 模式 ,可 能 的 取 值 包括 : RINGER_MODE_NORMAL 
(普通 模式 ) .RINGER_MODE_SILENT( 静 音 模式 ) .RINGER_MODE_VIBRATE( 振 动 模式 ) 等 。 此 
实例 的 完整 项 目 在 MyCode\ MySample066 文件 夹 中 ， 


文件 和 数据 


199 ”使 用 JSONObject 解析 JSON 字符 串 


此 实例 主要 通过 使 用 JSONTokener 和 JSONObject, 实 现 解析 JSON 字符 串 的 单个 对 象 。 当 实 
例 运行 之 后 , 单 击 “ 使 用 JSONObiject 解析 JSON 字符 串 的 单个 对 象 " 按 钮 , 则 将 把 JSON 字符 串 : 
{"myName":"Karli Watson", "myBook" :"C 井 入 门 经 典 "} ,解析 成 如 图 199. 1 所 示 的 效果 。 


个 “| 中 dx 村 “站 100% 上 和 


MySample 


使 用 JSONObject 解 析 JSON 字 符 串 的 单个 对 象 


作者 : Karli Watson， 书 名 :C# 入 门 经 典 


图 199.1 
主要 代码 如 下 : 


// 响 应 单 击 " 使 用 JSONObject 解析 JSON 字符 串 的 单个 对 象 "按钮 
public void onClickmyBtn]l (View v) { 
String myJsonData = "{\"myName\":\"Karli Watson\",\"myBook\":\"C# 入门 经 典 \"}"; 
try { 
String myInfo= ""; 
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JSONTokener myJSONTokener = new JSONTokener(myJsonData); 
JSONObject myJSONObject = (JSONObject) myJSONTokener. nextValue( ); 
myInfo+=" 作 者 :" + myJSONObject. getString("myName" ); 
myInfo+=", 书 名 :" + myJSONObject. getString("myBook" ); 
Toast. makeText (getApplicationContext( ), 
myInfo, Toast.LENGTH SHORT). show(); 
} catch (Exception e) { 
Toast. makeText( getApplicationContext(), 
e. getMessage( ) .toString(), Toast. LENGTH SHORT). show( ) ; 
} } 


上 面 这 段 代 人 码 在 MyCode\ MySample472\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,JSONTokener myJSONTokener 三 new JSONTokener 
CmyJsonData ) 用 于 根据 JSON 格式 的 字符 串 构 造 一 个 JSONTokener 实例 。JSONObject 
myJSONObject 二 (JSONObject) myJSONTokener. nextValue() 用 于 将 JSON 字符 串 唯一 的 对 象 解 
析 为 JSONObject 的 实例 。myJSONObject. getString("myName") 用 于 从 JSON 对 象 中 获取 属性 名 
称 为 myName 的 值 。 此 实例 的 完整 项 目 在 MyCode\MySample472 文件 夹 中 。 


200 “使 用 JSONArray 解析 JSON 字符 串 


此 实例 主要 通过 使 用 JSONArray 和 JSONObject, 实 现 解析 JSON 字符 串 的 在 套 对 象 。 当 实例 
运行 之 后 , 单 击 “ 使 用 JSONArray 解析 JSON 字符 串 的 府 套 对 象 " 按 钮 , 则 将 把 JSON 字符 串 : 
[{"myName":" Karli Watson"," myBook": (1 "bookName":"C 井 和 人 入门 经 典 ","bookPrice": 108， 
"bookStore" :" 京 东 商 城 ")),{( "myName":"Bruce Eckel","myBook": {( "bookName":"Java 编程 思 
想 " ," bookPrice": 78, "bookStore" :" 天 猫 商 城 "), ("myName":" Stephen Prata"," myBook": 
{"bookName" :"C++Primer Plus" , "bookPrice" :95,"bookStore" : "当当 商城 "}) ,解析 成 如 图 200. 1 
所 示 的 效果 。 

外山 


MySample 


使 用 JSONArray 解 析 JSON 字 符 串 的 谋 套 对 象 


作者 : Karli Watson， 出 版 专著 信息 : 书 
名 : C# 入 门 经 典 ,单价 : 108, 代 理 书 
店 : 京东 商城 

作者 : Bruce Eckel， 出 版 专著 信息 :, 书 
名 : Java 编 程 思想 ,单价 : 78, 代 理 书 
店 起 


一 -一 
Fr 二- 


了 击 。 大 针 隅 居 

作者 : Stephen Prata， 出 版 专著 信息 ; 

书 名 : C++ Primer Plus 单价 : 95 代理 书 
er fe 


占 : 当当 商城 


图 200. 1 
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主要 代码 如 下 : 


// 响 应 单 击 "使 用 JSONArray 解析 JSON 字符 串 的 从 套 对 象 "按钮 
public void onClickmyBtnl (View v) { 
String myJsonData = "[{\"myName\":\"Karli Watson\",\"myBook\":{\"bookName\": \"C# 入 门 经 典 \",\" 
bookPrice\" :108,\"bookStore\" :\" 京 东 商 城 \" }}，" 
+ "{\"myName\" :\"Bruce Eckel\",\" myBook\" : {\" bookName\" : \" Java 编程 思想 \"，\"bookPrice\" : 
78,\"bookStore\":\" 天 猫 商 城 \"}}," 
+"{\"myName\":\"Stephen Prata\",\"myBook\":{\"bookName\":\"Ct+Primer Plus\",\"bookPrice\": 
95, \"bookStore\" :\" 当当 商城 \"}}”+"]"; 
try { 
String myInfo=""; 
JSONArray myArray = new JSONArray(myJsonData); 
int myLength = myArray. length( ); 
for (int i = 0; i< myLength; i++) { 
JSONObject myJSONObject = myArray.getJSONObject(i); 
myInfo += "\n 作者 :" + myJSONObject. getString("myName" ); 
myInfo += ", 出 版 专著 信息 :”; 
JSONObject myBooks = myJSONObject. getJSONObject("myBook" ); 
myInfo += ", 书 名 :" + myBooks.getString("bookName" ); 
//myInfo +=", 单 价 :" + myBooks.getInt("bookPrice" ); 
myInfo += ", 单 价 :" + myBooks.getString("bookPrice" ); 
myInfo += ", 代理 书店 :" + myBooks. getString("bookStore" ); 
} 
Toast. makeText (getApplicationContext( ) ， 
myInfo，Toast. LENGTH SHORT) . show( ) ; 
} catch (Exception e) { 
Toast. makeText( getRApplicationContext( ) ， 
e. getMessage( ) .toString( ), Toast. LENGTH _ SHORT) . show( ) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySample476\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,{( "myName" :"Stephen Prata" , "myBook" : { "bookName" 
:"C++Primer Plus" ,"bookPrice" :95,"bookStore":" 当 当 商 城 ")) 有 两 对 大 括号 “{...{...)})”, 里 面 的 
大 括号 包含 的 对 象 是 外 面 的 大 括号 对 象 的 子 对 象 ,因此 两 个 对 象 在 解析 时 有 差异 ,myJSONObject = 
myArray. getJSONObject(D) 是 根据 数组 索引 1 解析 的 外 面 大 括号 对 象 ,myBooks 一 myJSONObject. 
getJSONObject("myBook") 是 根据 子 对 象 名 称 myBook 解析 的 里 面 大 括号 子 对 象 。 此 实例 的 完整 项 
日 在 MyCode\MySample476 文件 夹 中 。 


201 ”使 用 JSONTokener 解析 JSON 字符 串 


此 实例 主要 通过 使 用 JSONTokener 的 nextString() 方 法 ,从 而 获取 在 JSON 字符 串 中 指定 字符 
之 前 的 部 分 文本 。 当 实例 运行 之 后 , 单 击 “使 用 JSONTokener 获取 JSON 的 部 分 文本 ”按钮 , 则 在 弹 
出 的 Toast 中 显示 “大 家 系列 :{"myName":"Karli Watson","myBook":"C# 人 和信 门 经 由 ";” 字 符 串 的 
“和 人 ”字符 之 前 的 所 有 文本 ,如 图 201. 1 所 示 。 
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要 


MySample 


使 用 JSONTokener 获 取 JSON 的 部 分 文本 


大 家 系列 :{"myName""Karli 
Watson , myBook : C# 


图 201. 1 
主要 代码 如 下 : 


// 响 应 单 击 "使 用 JSONTokener 获取 JSON 的 部 分 文本 "按钮 
public void onClickmyBtnl (View v) { 
String myJsonData = 
"大 家 系列 :{\"myName\":\"Karli Watson\",\"myBook\":\"C# 入 门 经 典 \"}"; 
try { 
String myInfo= ""; 
JSONTokener myJSONTokener = new JSONTokener(myJsonData); 
myInfo + = myJSONTokener. nextString( ' 入 '); 
Toast. makeText (getApplicationContext(), 
myInfo, Toast.LENGTH SHORT). show( ); 
} catch (Exception e) { 
Toast. makeText (getApplicationContext( ), 
e. getMessage( ) .toString( ) ,Toast. LENGTH SHORT). show( ); 


y 


上 面 这 上段 代码 在 MyCode\ MySample478\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,myJSONTokener. nextString(' 和 信 ') 用 于 获取 “大 家 系列 : 
{"myName":"Karli Watson","myBook":"C# 和 人 入门 经 肉 "}” 字 符 串 的 “入 ”字符 之 前 的 所 有 文本 “大 
家 系列 : {1" myName":" Karli Watson"," myBook":"C#”。 此 实例 的 完整 项 目 在 MyCode\ 
MySample478 文件 夹 中 。 


202 ”使 用 JsonReader 解析 JSON 字符 串 


此 实例 主要 通过 使 用 JsonReader 的 beginArray() 、beginObject() nextName() 和 nextString( ) 
等 方法 ,实现 将 JSON 格式 的 字符 串 以 对 象 的 方式 解析 出 来 。 当 实例 运行 之 后 , 单 击 “使 用 
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JsonReader 解析 JSON 字符 串 ” 按 钮 , 则 将 把 JSON 字符 串 : [1"myName":"Karli Watson "， 
"myBook":"C# 人 入门 经 典 "},{"myName":" Bruce Eckel",，" myBook":" Java 编程 思想 "}， 
{"myName":"Stephen Prata","myBook" :"C++Primer Plus")] ,解析 成 如 图 202. 1 所 示 的 效果 。 


| Ww Q = 中 是 xx 


MySample 


使 用 JsonReader 解 析 JSON 字 符 串 


;Karli Watson , 书 名 : C# 入 门 经 典 
” Bruce Eckel, 书 名 : Java 编 程 思想 
: Stephen Prata, 书 名 : C++ Primer 


图 202. 1 
主要 代码 如 下 : 


// 响 应 单 击 " 使 用 JsonReader 解析 JSON 字符 串 " 按 钮 
public void onClickmyBtn]l (View v) { 
String myJsonData = 
"[{\"myName\" :\"Karli Watson\",\"myBook\":\"C# 入 门 经 典 \"}," + 
"{\"myName\" :\" Bruce Eckel\",\"myBook\" :\"Java 编程 思想 \"}," + 
"{\"myName\" :\" Stephen Prata\",\"myBook\":\"Ct+Primer Plus\"}]"; 
| 
JsonReader myJsonReader = new JsonReader(new StringReader(myJsonData)); 
myJsonReader. beginArray( ); 
String myInfo = ""; 
while (myJsonReader. hasNext()) { 
myJsonReader. beginObject( ); 
while (myJsonReader. hasNext()) { 
String myTagName = myJsonReader. nextName( ); 
if (myTagName. equals("myName" ) ) { 
myInfo += "\n 作者 :" + myJsonReader. nextString( ); 
} else if (myTagName.equals("myBook")) { 
myInfo += ", 书 名 :" + myJsonReader. nextString(); 
} } 
myJsonReader. endObject( ); 
} 
myJsonReader. endArray( ); 
Toast. makeText (getApplicationContext(), myInfo, Toast.LENGTH SHORT). show(); 
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} catch (Exception e) { 
Toast. makeText( getApplicationContext( ), 
e. getMessage( ) .toString( ) ，Toast. LENGTH SHORT). show( ) ; 
3 


上 面 这 段 代 码 在 MyCode\ MySample468\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文 件 中 。 在 这 段 代 码 中 ， myjsonReader = new JsonReader (new StringReader 
(myJsonData) ) 用 于 根据 JSON 字符 串 构 造 JsonReader 实例 。 myJsonReader. beginArray ( ) 和 
myJsonReader. beginObject() 表 示 JsonReader 开始 执行 解析 动作 。myJsonReader. nextName() 用 于 
获取 JSON 标签 名 称 。myJsonReader. nextString() 用 于 获取 革 个 标签 对 应 的 内 容 , 标 签 和 内 容 之 间 
用 冒号 连接 ,就 像 是 一 个 键 值 对 。 此 实例 的 完整 项 目 在 MyCode\MySample468 文件 夹 中 。 


203 ”使 用 JSONStringer 创建 JSON 字符 串 


此 实例 主要 通过 使 用 JSONStringer, 创建 JSON 格式 的 字符 串 。 当 实例 运行 之 后 , 单 击 * 使 用 
JSONStringer 创建 JSON 格式 的 字符 串 ” 按 钮 , 则 将 创建 包含 三 个 对 象 的 数组 型 的 JSON 字符 串 , 如 
图 203. 1 的 Toast 所 示 。 

主要 代码 如 下 : 


// 响 应 单 击 " 使 用 JSONStringer 创建 JSON 格式 的 字符 串 "按钮 外 QOrY 
public void onClickmyBtn]l (View v) { 
try { 

JSONStringer myJSONStringer = new JSONStringer(); 使 用 JSONStringer 创 建 JSON 格 式 的 字符 串 
myJSONStringer. array( ); 
myJSONStringer. object( ); 
myJSONStringer. key(" myName" ); 
myJSONStringer. value(" Karli Watson" ); 
myJSONStringer. key(" myBook" ); 
myJSONStringer. value("C# 入 门 经 典 " ); 
myJSONStringer. endObject( ); 
myJSONStringer. object( ); 
myJSONStringer. key(" myName" ); 
myJSONStringer. value(" Bruce Eckel" ); 
myJSONStringer. key(" myBook" ); 


MySample 


四 m [fmyName" :Karli Watsonm “myBook":"C# 
myJSONStringer. Value( Java 编程 思想 ); 入 i ] 经 典 }),{'myName ‘Bruce 
myJSONStringer. endObject( ); Eckel"“myBook"Java 编 程 思 想 ), 

i {myName": Stephen Prata ,myBook C++ 
myJSONStringer. object( ); Primer Plus 


myJSONStringer. key(" myName" ); 
myJSONStringer. value(" Stephen Prata" ); 
myJSONStringer. key(" myBook ); 
myJSONStringer. value("C++Primer Plus" ); 图 203.1 
myJSONStringer. endObject( ); 
myJSONStringer. endArray( ); 
Toast. makeText (getApplicationContext(), 

myJSONStringer. toString(), Toast. LENGTH SHORT). show( ) ; 

} catch (Exception e) { 

Toast. makeText( getApplicationContext( ), 

e. getMessage( ). toString(), Toast. LENGTH SHORT). show( ) ; 


1 


上 F 面 这 段 代 码 在 MyCode\ MySample474\app\src\ main\java\com\bin\luo\mysample\ 
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MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myJSONStringer. key("myName") 用 于 创建 键 名 
myName,myJSONStringer. value("Stephen Prata") 用 于 创建 键 值 Stephen Prata, 在 JSON 字符 串 
中 , 键 名 和 键 值 用 “:” 连 接 , 每 个 键 名 键 值 对 之 间 用 “,” 连 接 。myJSONStringer. object ( ) 和 
myJSONStringer. endObject() 用 于 构造 一 个 对 象 ,相当 于 JSON 字符 串 的 “{17 和 ”)”, 每 个 对 象 之 间 也 
用 “,” 连 接 。myJSONStringer. array() 和 myJSONStringer. endArray() 用 于 构造 对 象 数组 ,相当 于 
JSON 字符 串 的 “[” 和 “jj]”。myJSONStringer. toString() 表 示 以 JSON 格式 输出 myJSONStringer 的 
内 容 。 此 实例 的 完整 项 目 在 MyCode\MySample474 文件 夹 中 。 


204 使 用 JSONObject 根据 IP 显示 所 在 城市 


此 实例 主要 通过 使 用 JSONObject 解析 从 http://int. dpool. sina. com. cn 返回 的 信息 ,获取 本 机 
IP 对 应 的 城市 信息 。 当 实例 运行 之 后 , 单 击 “ 查 询 本 机 IP 地 址 ”按钮 , 则 在 弹出 的 Toast 中 显示 本 机 
IP 信息 ,如 图 204. 1 的 左 图 所 示 ; 单 击 “查询 本 机 位 置 ” 按 钮 , 则 在 弹出 的 Toast 中 显示 本 机 对 应 的 城 
市 信息 ,如 图 204. 1 的 右 图 所 示 。 


权 帮 分 和 苗 中 x 志 下 午 2: 万 分 加 四 日 


MySample MySample 


查询 本 机 IP 地 址 查询 本 机 位 置 查询 本 机 IP 地 址 查询 本 机 位 置 


本 机 IP 地 址 是 : 192.168.1.102 本 机 位 置 在 : 中 国 重庆 重庆 


图 204. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
Handler myHandler = new Handler() { 
(Override 
public void handleMessage( Message msg) { 
String myData = (String) msg. obj; 
try { 
JSONObject myJSONObject = 
new JSONObject (myData. substring(myData. indexOf(" {" ))); 
String myCity = (String) myJSONObject.get("city" ); 
String myCoutry = (String) myJSONObject.get("country" ); 
String myProvince = (String) myJSONObject.get("province" ); 
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Toast. makeText (getApplicationContext(), "本 机 位 置 在 :" + myCoutry + "" 
+ myProvince + " " + myCity + "市 ", Toast.LENGTH LONG). show(); 
} catch (Exception e) { 
Toast. makeText (getApplicationContext(), 
"查询 失败 !" ，Toast.LENGTH_SHORT). show( ); 
} 
private String myIPAddress; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
} 
public void onClickBtnl (View view) { // 响 应 单 击 "查询 本 机 IP 地 址 "按钮 
ConnectivityManager myManager = 
(ConnectivityManager) getSystemService(Context.CONNECTIVITY SERVICE); 
NetworkInfo myMobileNetInfo = 
myManager. getNetworkInfo(ConnectivityManager. TYPE MOBILE ) ; 
NetworkInfo myWifiNetInfo = 
myManager. getNetworkInfo(ConnectivityManager. TYPE WIFI); 
if (!myMobileNetInfo. isConnected() && !myWifiNetInfo. isConnected()) { 
Toast. makeText (MainActivity. this, 
"当前 网 络 不 可 用 !"，Toast. LENGTH_SHORT). show( ); 
} else if (myMobileNetInfo. isConnected( ) && !myWifiNetInfo. isConnected( )) { 
myIPAddress = getIPAddressByMobileNet( ) ; 
} else if (!myMobileNetInfo. isConnected() && myWifiNetInfo. isConnected()) { 
WifiManager myWifiManager = 
(WifiManager) getSystemService(Context. WIFI SERVICE); 
WifiInfo myWifiInfo = myWifiManager.getConnectionInfo( ); 
myIPAddress = FormatIPAddress(myWifiInfo.getIpAddress( )); 
} 
Toast. makeText (getApplicationContext( ), 
"本 机 IP 地 址 是 :" + myIPAddress, Toast.LENGTH LONG). show( ); 
} 
// 响 应 单 击 "查询 本 机 位 置 "按钮 
public void onClickBtn2(View view) { getLocalInfo(); } 
public static String FormatIPAddress( int ipAddress) { 
StringBuilder myBuilder = new StringBuilder( ); 
myBuilder. append( ipAddress & OxFF).append("."); 
myBuilder. append( (ipAddress >> 8) & OxFF).append("."); 
myBuilder. append( ( ipAddress >> 16) & OxFF).append("."); 
myBuilder.append( (ipAddress >> 24) & OxFF); 
return myBuilder. toString( ); 
} 
public void getLocalInfo() { 
new Thread( ) { 
@ Override 
public void run() { 
try { 
URL myUrl = 
new URL("http://int. dpool. sina. com. cn/ iplookup/ iplookup. php?format = js" ); 
URLConnection myConnection = myUrl.openConnection( ); 
BufferedReader myReader = new BufferedReader( 
new InputStreamReader (myConnection. getInputStream( ), "GBK" )); 
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String myLength = null; 
StringBuffer myBuffer = new StringBuffer(); 
while ((myLength = myReader.readLine()) != null) { 
myBuffer. append( myLength); 
} 
myReader. close( ); 
String myResult = myBuffer.toString( ); 
Message myMessage = new Message(); 
myMessage. obj = myResult; 
myHandler. sendMessage( myMessage); 
} catch (Exception e) { } } 
}. start(); 
} 
public static String getIPAddressByMobileNet() { 
try { 
for (Enumeration < NetworkInterface> myEnum = 
NetworkInterface. getNetworkInterfaces(); myEnum. hasMoreElements(); ) { 
NetworkInterface myNetInterface = myEnum. nextElement( ) ; 
for (Enumeration < InetAddress > myIPAddress = 
myNetInterface. getInetAddresses( ) ; myIPAddress. hasMoreElements(); ) { 
InetAddress myAddress = myIPAddress. nextElement( ); 
if (!myAddress. isLoopbackAddress( )&& myAddress instanceof Inet4Address) { 
return myAddress. getHostAddress( ). toString( ); 
汗 圭 ， 
} catch (Exception e) { 
} 
return null; 


前 


上 面 这 段 代 码 在 MyCode\MySample740\app\src\main\java\com\bin\、 em 


1, start :-1,end :-1,country : \u4e2d 


luo\ mysample\ MainActivity. java 文件 中 。 在 这 段 代码 中 ,myUrl = new 
URL("http://int. dpool. sina. com. cn/iplookup/iplookup. php? format= 
js") 用 于 创建 URL ,该 URL 在 网 络 连接 成 功 ( 有 可 能 失败 ) 之 后 ,将 以 JSON 
字符 串 的 形式 返回 查询 者 本 号 的 位 置信 息 , 如 图 204. 2 所 示 。 当 使 用 
myJSONObject 三 new JSONObject(myData. substring(myData. indexOf("{"))) 构 造 JSON 对 象 之 
后 , 即 可 用 myCity 二 (String) myJSONObject. get("city" ) 的 形式 进行 查询 。 

此 外 ,访问 网 络 需 要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name = 


"android. permission. ACCESS_ WIFI_ STATE"/>、< uses-permission android: name = "android. 


permission. INTERNET"/> 和 < uses-permission android: name = "android. permission. ACCESS _ 


NETWORK_STATE"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample740 文件 夹 中 ， 


205 ”使 用 Gson 将 数组 转换 成 JSON 字符 串 


此 实例 主要 通过 使 用 Gson 的 toJson() 方 法 ,实现 将 数组 对 象 转换 成 JSON 字符 串 。 当 实例 运行 
之 后 , 单 击 “使 用 Gson 将 数组 对 象 转换 成 JSON 字符 串 ” 按 钮 ,将 把 包含 三 个 对 象 的 数组 解析 成 如 
图 205. 1 所 示 的 JSON 字符 串 。 
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主要 代码 如 下 : 
public class MainActivity extends Activity { 外 器 中 因 d* 写 
(@Override ES 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 使 用 Gson 将 数组 对 象 转换 成 JSON 字 符 串 
setContentView(R. layout.activity main); 
} 


// 响 应 单 击 "使 用 Gson 将 数组 对 象 转换 成 JSON 字符 串 "按钮 
public void onClickmyBtnl (View v) { 

// 创 建 数组 对 象 并 添加 数据 

ArrayList <User> myArray = new ArrayList <>(); 

User myUserl = new User(); 

myUser1. setMyName(" Karli Watson" ); 

myUser1. setMyBook("C# 入 门 经 典 " ); 

User myUser2 = new User(); 

myUser2. setMyName(" Bruce Eckel" ); 

myUser2. setMyBook(" Java 编程 思想 " ); A A eT 


Watson"}.{"myBook”…Java 编 程 思 


nd ER 
myUser3. setMyName(" Stephen Prata" ) ; prata")] i 和 
myUser3. setMyBook("C++Primer Plus" ); 
myRrray. add(myUser1l ); 
myRrray. add(myUser2 ) ; 
myArray.add(myUser3); 各 ”205.1 
try 

Gson myGson = new Gson( ); 

String myInfo = myGson. toJson(myArray); 

Toast. makeText (getApplicationContext(), myInfo, Toast. LENGTH SHORT). show( ) ; 
} catch (Exception e) { 

Toast. makeText (getApplicationContext(), 

e. getMessage( ) .toString(), Toast. LENGTH SHORT). show( ) ; 


时 | 

class User{ //User 类 定义 
public String getMyName( ) { return myName; } 
public void setMyName( String myName) { this.myName = myName; } 
public String getMyBook( ) { return myBook; } 
public void setMyBook( String myBook) { this.myBook 
String myName; 
String myBook; 

最 


myBook; } 


上 F 面 这 段 代 码 在 MyCode\ MySample471 \app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mylInfo= 二 myGson. toJson(myArray) 用 于 将 myArray 数 
组 中 的 三 个 对 象 以 标准 格式 的 JSON 字符 串 输 出 到 myInfo 中 。 此 外 ,在 Android Studio 中 使 用 Gson 
需要 在 gradle 中 引入 compile 'com. google. code. gson:gson:2. 8.1' 依 赖 项 。 此 实例 的 完整 项 目 在 
MyCode\MySample471 文件 夹 中 ， 


206 ”使 用 Gson 解析 JSON 字符 串 


此 实例 主要 通过 使 用 Gson 的 fromJson() 方 法 ,实现 将 JSON 格式 的 字符 串 以 对 象 的 方式 解析 出 

来 。 当 实例 运行 之 后 , 单 击 “ 使 用 Gson 解析 JSON 格式 的 字符 串 ” 按 钮 , 则 将 把 JSON 字符 串 : 
{"myName":" Karli Watson"," myBook":"C# 入门 经 典 "}, {" myName":" Bruce Eckel"," 
myBook" :"Java 编程 思想 "},{"myName": "Stephen Prata","myBook":"C++Primer Plus") |] ,解析 
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成 如 图 206. 1 所 示 的 效果 。 
a ~ 中国 dx 环 ”dd| 站 100% 上 午 11:22 


MySample 


使 用 Gson 解 析 JSON 格 式 的 字符 串 


作者 : Karli Watson， 书 名 : C# 入 门 经 
作者 : Bruce Eckel， 

思想 

作者 : Stephen Prata， 书 名 : C++ 
Primer Plus 


图 206.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 
} 
// 响 应 单 击 " 使 用 Gson 解析 JSON 格式 的 字符 串 "按钮 
public void onClickmyBtnl (View v) { 
String myJsonData = "[{\"myName\":\"Karli Watson\", \"myBook\":\"C# 人 入门 经 典 \"}," 
+ "{\"myName\":\"Bruce Eckel\", \"myBook\" :\"Java 编程 思想 \"}," 
+"{\"myName\":\"Stephen Prata\",\"myBook\":\"C++Primer Plus\"}]"; 


try { 
Type myType = new TypeToken < LinkedList <User >>(){}.getType( ); 
Gson myGson = new Gson( ); 


LinkedList < User > myUsers = myGson.fromJson(myJsonData, myType); 
String myInfo=""; 
for (Iterator myIterator = myUsers. iterator(); myIterator.hasNext();) { 
User myUser = (User) myIterator. next( ); 
myInfo += "\n 作者 :" + myUser.getMyName( ); 
myInfo += "， 书 名 :”+ myUser. getMyBook( ); 
} 
Toast. makeText (getApplicationContext(), myInfo, Toast.LENGTH SHORT) . show( ) ; 
} catch (Exception e) { 
Toast. makeText (getApplicationContext(), 
e. getMessage( ) .toString(), Toast.LENGTH SHORT). show( ) ; 
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//User 类 定义 
class User{ 

public String getMyName() { return myName; } 
public void setMyName( String myName) { this.myName = myName; } 
public String getMyBook() { return myBook; } 
public void setMyBook( String myBook) { this.myBook 
String myName; 
String myBook; 

Es 


myBook; } 


上 面 这 段 代 码 在 MyCode\ MySample469\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myType 二 new TypeToken < LinkedList < User >>() 
{}. getType( ) 用 于 获取 自 定 义 类 User 的 类 型 定义 。LinkedList < User > myUsers 一 myGson. 
fromJson(myJsonData, myType) 用 于 根据 User 的 类 型 定义 myType 和 Json 字符 串 myJsonData 的 
内 容 获 取 对 象 集 合 。myIterator 一 myUsers. iterator( ) 用 于 获取 集合 枚 举 器 。myUser 一 (User) 
mylterator. next() 用 于 从 集 合 枚 举 器 中 获取 单个 对 象 。 此 外 ,在 Android Studio 中 使 用 Gson 需要 在 
gradle 中 引入 compile 'com. google. code. gson:gson:2.8.1' 依 赖 项 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample469 文件 夹 中 ， 


207 ”使 用 XmlPullParser 解析 城市 天 气 数据 


此 实例 主要 通过 调用 聚合 数据 提供 的 天 气 预报 API 接口 ,查询 指定 城市 的 天 气 信息 并 返回 XML 
格式 数据 ,然后 使 用 XmlPullParser 解析 这 些 城市 的 天 气 数据 ,从 而 获取 指定 城市 的 天 气 信息 。 当 实 
例 运行 之 后 ,在 “城市 : ”输入 框 中 输入 城市 名 称 , 如 “重庆 ”, 然 后 单 击 “ 查 询 当 地 今天 的 天 气 ” 按 钮 , 则 
在 下 面 显示 该 地 今天 的 天 气 信息 ; 输入 其 他 城市 名 称 查询 天 气 信 息 将 获得 相应 的 结果 ,效果 分 别 如 
图 207. 1 的 左 图 和 右 图 所 示 。 


外 0 吕 “~ 中 因由 


MySample MySample 


城市 : 重庆 查询 当地 今天 的 天 气 “ | 城市: 武汉 查询 当地 今天 的 天 气 
2018 年 06 月 20 日 2018 年 06 月 20 日 


24C~29C 2 ti G 


北 风 3-5 级 


市 大 学 的 了 站 理工 


人 8》 Anaroia 红 应 用 200 。 实战 篇 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
EditText myEditCity; 
Handler myHandler = new Handler() { 
Override 
public void handleMessage(Message msg) { 
String myData = (String) msg.obj; 
if (msg.what == 2) { 


// 将 字符 串 转换 为 输入 流 形式 
InputStream myInputStream = new ByteArrayInputStream(myData. getBytes( )); 
ParseWeatherInfo(myInputStream) ; // 使 用 XmlPullParser 对 XML 数据 进行 解析 


myTemperature. setText (myWeatherList. get(0).TemperatureInfo); 
myWind. setText (myWeatherList. get(0).WindInfo); 
myWeatherText. setText (myWeatherList. get(0).WeatherInfo); 
myLayout. setVisibility(View. VISIBLE); // 显 示 天 气 信 息 
myLoading. setVisibility(GONE); 
}}}; 
private ArrayList < Weather > myWeatherList; 
private Weather myWeather; 
private TextView myTemperature; 
private TextView myWind; 
private TextView myWeatherText; 
private TextView myLoading; 
private LinearLayout myLayout; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myEditCity = (EditText) findViewById(R. id.myEditCity); 
myTemperature = (TextView) findViewById(R. id.myTemperature); 
myWind = (TextView) findViewById(R. id.myWind); 
myWeatherText = (TextView) findViewById(R. id.myWeather); 
myLoading = (TextView) findViewById(R. id. myLoading) ; 
myLayout = (LinearLayout) findViewById(R. id.myLayout); 
} 
public void onClickmyBtnl (View v) { // 响 应 单 击 "查询 当地 今天 的 天 气 " 按 钮 
myLayout. setVisibility(GONE); 
myLoading. setVisibility(View. VISIBLE) ; 
getWeatherInfo(myEditCity. getText( ) .toString( ) ) ; 
} 
// 这 里 使 用 的 是 聚合 数据 的 API 接口 ,调用 时 请 更 换 为 自己 申请 的 APPKEY 值 
public void getWeatherInfo(final String myCityName) { 
new Thread( ) { 
(Override 
public void run() { 
try { 
URL myUrl = new URL("http://v. juhe. cn/weather/ index?format = 2&cityname = ”+ 
myCityName + "Skey = 94e8b799a9d12a50bacb61c5282e603egdtype = xml" ); 
URLConnection myConnection = myUrl.openConnection( ) ; 
BufferedReader myReader = new BufferedReader(new InputStreamReader( 
myConnection. getInputStream( ), "UTF - 8" )); 
String myLength = null; 
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StringBuffer myBuffer = new StringBuffer( ); 


while ((myLength = myReader.readLine())!= null){ myBuffer.append(myLength);} 
myReader. close( ); 
String myResult = myBuffer.toString( ); 
Message myMessage = new Messagel( ); 
myMessage. obj = myResult; 
myMessage.what = 2; 
myHandler. sendMessage( myMessage); 
} catch (Exception e) { e.printStackTrace( ); } 
} 
1 atart(}: 
} 
public void ParseWeatherInfo( InputStream inputStream) { 


try { 
XmlPullParser myParser = Xml.newPullParser(); 
myParser. setInput (inputStream, "UTF - 8" ); // 传 入 需要 解析 的 XML 数据 流 
int myEvent = myParser.getEventType( ); // 获 取 事 件 类 型 


while (myEvent != XmlPullParser. END _ DOCUMENT) { 
String myNodeName = myParser.getName( ); 
switch (myEvent) { 


case XmlPullParser. START DOCUMENT: // 文 档 开 始 
myWeatherList = new ArrayList < Weather >(); 
break; 

case XmlPullParser. START TAG: // 标 签 开 始 


if ("today" .equals(myNodeName)) { myWeather = new Weather(); } 

if ("temperature" .equals(myNodeName)) { 
String myTemperatureInfo = myParser.nextText(); 
myWeather. setTemperatureInfo(myTemperatureInfo); 

} 

if ("weather" .equals(myNodeName)) { 
String myWeatherInfo = myParser.nextText( ); 
myWeather. setWeatherInfo(myWeatherInfo); 

} 

if ("wind" .equals(myNodeName)) { 
String myWindInfo = myParser.nextText(); 
myWeather. setWindInfo(myWindInfo); 

} 

break; 

case XmlPullParser. END TAG: // 标 签 结 束 

if ("today" .equals(myNodeName)) { 
myWeatherList. add(myWeather ); 
myWeather = null; 

} 

break; 

} 
myEvent = myParser. next(); // 下 一 个 标签 
} 
} catch (Exception e) { 
e. printStackTrace( ) ; 
上 


上 上面 这 段 代 码 在 MyCode\ MySample753\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myData = (String) msg. obj 返回 的 天 气 信 息 原 本 是 一 个 
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XML 格式 的 文本 ,如 下 所 示 : 


<?xml version= "1.0" encoding = "utf 一 8"?> 

< root> 

< resultcode> 200 </resultcode > 

< reason > successed!</reason> 

< reSsult>< sk >< temp > 9 </temp > 

< Wind direction> 西 北 风 </wind direction > 

< wind strength> 2 级 </wind_ strength > 

< humidity> 95 % </humidity> 

<time> 14:01 </time></sk> 

< today>< temperature > 9T 一 11YC </temperature > 

< Weather > 小 十 </weather > 

< weather_id>< fa> 07 </fa>< fb> 07 </fb></weather_id > 
< wind > 持续 无 风向 微风 </wind > 

< week > 星期 三 </week > 

< city> 重 庆 </city> 

< date y>2018 年 02 月 21 日 </date y> 

<dressing index > 较 冷 </dressing index> 

< dressing_advice> 建 议 着 厚 外 套 加 毛衣 等 服装 .年 老 体 弱者 宜 着 大 衣 、 呢 外 套 加 羊毛 衫 . </dressing_advice> 
< uv_index > 最 弱 </uv_index > 

<comfort index></comfort index> 

<wash index > 不宜 </wash_index> 

<travel index> 较 不 宜 </travel_index> 

< exercise index > 较 不 宜 </exercise_ index> 

<drying index></drying index ></today></result >< error code> 0 </error code></root> 


ParseWeatherInfo() 方 法 则 使 用 XmlPullParser 根据 上 述 文 本 的 各 个 XML 标签 进行 指定 内 容 的 
解析 ,Pull 解析 XML 是 基于 事件 驱动 方式 的 解析 ,Pull 开始 解析 时 ,可 以 先 通 过 getEventType() 方 
法 获取 当前 解析 事件 类 型 ,并 且 通 过 next() 方 法 获取 下 一 个 解析 事件 类 型 。Pull 解析 费 提 供 了 
START_DOCUMENT( 开 始 文档 )\END_DOCUMENT (结束 文档 )\START_TAG (开始 标 签 )、 
END_TAG( 结 束 标签 ) 四 种 事件 解析 类 型 。 当 处 于 某 个 元 素 时 ,可 以 调用 getAttributeValue() 方 法 
获取 属性 的 值 , 也 可 以 通过 nextText() 方 法 获取 本 节点 的 文本 值 。 

此 外 ,访问 网 络 需 要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 一 
"android. permission. INTERNET"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample753 文件 夹 中 ， 


208 采用 SAX 方式 解析 XML 文件 内 容 


此 实例 主要 通过 使 用 SAXParserFactory 等 SAX 操作 类 ,实现 以 对 象 的 形式 读 取 XML 文件 的 内 
容 。 当 实例 运行 后 , 单 击 “采用 SAX 方式 解析 XML 文件 内 容 ” 按 钮 , 则 在 ListView 控件 中 显示 
personsax. xml 文件 的 内 容 , 效 果 分 别 如 图 208. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickmyBtn1 (View v) { // 响 应 单 击 " 采 用 SAX 方式 解析 XML 文件 内 容 "按钮 
try { 
myArray = ReadXMLFileBySAX(); 
} catch (Exception e) { e.printStackTrace( ); } 
myAdapter = new ArrayAdapter < Person>(MainActivity.this, 
android. R. layout. simple expandable list item 1, myArray); 
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myListView. setAdapter(myAdapter ); 


} 

private ArrayList < Person > ReadXMLFileBySAX() throws Exception { 
InputStream myInputStream = 

getAssets().open(" personsax. xml" ); // 根 据 XML 文件 创建 输入 流 对 象 
SaxHelper mySaxHelper = new SaxHelper(); // 创 建 xML 解析 处 理 器 
SAXParserFactory myFactory = 
SAXParserFactory. newInstance( ); // 得 到 SAX 解析 工厂 

SAXParser myParser = myFactory. newSRXParser( ); // 创 建 SAX 解析 器 
myParser. parse(myInputStream, mySaxHelper); // 对 文档 进行 解析 
myInputStream. close( ); 
return mySaxHelper. getPersonArrayList( ); 

} 


Ob tO 0O0v dda” 


MySample MySample 


采用 SAX 方 式 解析 XML 文件 内 容 采用 SAX 方 式 解析 XML 文件 内 容 


姓名 : 米兰 达 . 卡 斯 格拉 夫 ， 年 龄 : 37 


姓名 : 迈克 尔 . 贝 亚 蒂 ， 年龄 : 45 


姓名 : 达 纳 . 盖 尔 ， 年 龄 : 29 


姓名 : 克 里 斯 汀 ` 韦 格 ， 年 龄 : 21 


图 208.1 


上 F 面 这 段 代 码 在 MyCode\ MySample332\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySaxHelper = new SaxHelper() 用 于 创建 SaxHelper 
类 的 实例 ,以 解析 XML 文件 的 标签 。 关 于 SaxHelper 类 的 定义 请 参考 源 代码 中 的 MyCode\ 
MySample332\app\src\main\java\com\bin\luo\mysample\SaxHelper. java 文件 。SAX ,全 称 Simple 
API for XML,SAX 不 同 于 DOM 解析 , 它 逐 行 扫 描 文 档 ,一边 扫 描 一 边 解 析 。 由 于 应 用 只 是 在 该 取 
时 检查 数据 ,因此 不 需要 将 全 部 数据 加 载 在 内 存 中 ,这 对 于 大 型 文档 的 解析 是 个 巨大 优势 。 此 实例 的 
完整 项 目 在 MyCode\MySample332 文件 夹 中 。 


209 使 用 Pattern 根据 正则 表达 式 校 验 手 机 号 码 


此 实例 主要 通过 使 用 Pattern 的 compile() 和 matcher() 方 法 ,实现 通过 正则 表达 式 校 验 宁 符 串 格 
式 的 手机 号 码 是 否 正确 。 当 实例 运行 之 后 ,如 果 在 “手机 号 码 : ”输入 框 中 输入 “13996060872-1”, 然 后 
单 击 “ 使 用 正则 表达 式 检验 手机 号 码 ” 按 钮 , 则 在 弹出 的 Toast 中 提示 此 号 码 错误 ,如 图 209. 1 的 左 图 
所 示 。 如 果 在 “手机 号 码 : ”输入 框 中 输入 “13996060872”, 然 后 单 击 “ 使 用 正则 表达 式 检验 手机 号 码 ” 
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按钮 , 则 在 弹出 的 Toast 中 提示 此 号 码 正确 ,如 图 209. 1 的 右 图 所 示 。 


MySample MySample 


手机 号 码 : 13996060872-1 于 机 号 码 ; 13996060872 


使 用 正则 表达 式 校 验 手 机 号 码 使 用 正则 表达 式 校 验 手 机 号 码 


13996060872-1 可 能 不 是 正确 的 手机 号 
码 ! 13996060872 是 正确 的 手机 号 人 码 ! 


图 209.1 
public void onClickmyBtnl (View v) { // 响 应 单 击 "使 用 正则 表达 式 校 验 电话 号 码 "按钮 
String myPhone = myEgditText. getText( ) .toString( ); 
if(!checkPhone(myPhone) ){ 


Toast. makeText (getApplicationContext( ), 
myPhone + " 权能 不 是 正确 的 手机 号 码 !"， Toast.LENGTH SHORT). show(); 
}else{ 
Toast. makeText (getApplicationContext(), 
myPhone + " 是 正确 的 手机 号 码 !" ，Toast.LENGTH SHORT). show( ); 
重 
public static boolean checkPhone(String myPhone){ // 判 断 最 新 的 中 国电 话 号 码 
Pattern myPattern = Pattern. compile( 
S030 II ULSITI (S00 = TLS= oI0 5 oT 9 
Pattern. CASE INSENSITIVE); 
Matcher myMatcher = myPattern.matcher(myPhone); 
return myMatcher. matches( ); 
} 


上 面 这 上段 代码 在 MyCode\ MySample372\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myPattern 一 Pattern. compile("^((13[0 一 91)| (14[5 | 
7])|1(15([0 一 3]1[5 一 9]))1(18[0,5 一 9]))NXd(8)$"，Pattern. CASE_ INSENSITIVE) 用 于 根据 字符 
串 指定 的 规则 创建 一 个 正则 表达 式 的 实例 ,当前 这 个 字符 串 主 要 用 于 匹配 中 国 大 陆地 区 的 手机 号 码 ， 
如 下 : 

* 移动 号 码 段 :139、138、137、136、135、134、150、151、152、157、158、159、182、183、187、188、147。 

x 联通 号 码 段 :130、131、132、136、185、186、145。 

x 电信 和 号码 段 :133、153、180、189。 

myMatcher 王 myPattern. matcher(myPhone) 用 于 校 验 myPhone 字符 串 是 否 符 合 myPattern 下 
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则 表达 式 的 要 求 , 如 果 myMatcher. matches() 返 回 值 为 true, 则 表示 符合 要 求 ; 否则 不 符合 要 求 。 此 
实例 的 完整 项 目 在 MyCode\MySample372 文件 夹 中 ， 


210 ”使 用 SharedPreferences 保存 账户 和 密码 


此 实例 主要 通过 使 用 SharedPreferences ,实现 以 键 值 对 的 形式 在 手机 中 保存 账户 和 密码 信息 。 
当 实 例 运 行 之 后 ,在 账户 和 密码 输入 框 中 输入 内 容 , 选 中 “ 记 住 账户 和 密码 信息 ” 复 选 框 ,然后 单 击 “ 登 
录 ” 按 钮 , 则 将 执行 登录 操作 ,并 将 账户 和 密码 信息 保存 在 手机 中 ,否则 提示 操作 错误 ,效果 分 别 如 
图 210. 1 的 左 图 和 右 图 所 示 。 在 Android 系统 中 ,SharedPreferences 保存 的 数据 文件 一 般 存 储 在 / 
data /data/com. bin. luo. mysample (your packagename )/shared _prefs 目录 下 。 此 实例 保存 的 
SharedPreferences 数据 文件 名 称 是 com. bin. luo. mysample_preferences. xml, 内 容 如 下 : 


记 住 账户 和 密码 信息 记 住 账户 和 密码 信息 


图 210.1 


<?xm] version = '1.0' encoding = 'utf ~ 8' standalone = 'yes' ?> 
<map>< string name = "account"> luobin </string > 
< String name = "password"> 123456 </string > 


< boolean name = "remember password" value = "true" /></map> 


该 功能 的 具体 实现 代码 如 下 : 


public class MainActivity extends Activity { 
public static final String REMEMBER PASSWORD = "remember password"; 
public static final String ACCOUNT = "account"; 
public static final String PASSWORD = "password"; 
private EditText myEditAccount, myEditPassword; 
private CheckBox myCheckbox; 
private SharedPreferences mySharedPreferences; 
private SharedPreferences. Editor myEditor; 
private boolean isRemember; 


办 
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(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout.activity main); 
myEditAccount = (EditText) findViewById(R. id.myAccount); 
myEditPassword = (EditText) findViewById(R. id.myPassword); 
myCheckbox = (CheckBox) findViewById(R. id.myCheckbox); 
// 获 取 SharedPreferences 对 象 
mySharedPreferences = PreferenceManager.getDefaultSharedPreferences(this); 
REMEMBER PASSWORD, false); 


if (isRemember) { 
// 将 SharedPreferences 保存 的 账户 和 密码 设置 到 TextView 中 
myEditAccount. setText (mySharedPreferences. getString(ACCOUNT, "" )); 
myEditPassword. setText (mySharedPreferences. getString(PASSWORD, "" )); 
myCheckbox. setChecked(true); 
}} 
public void onClickmyBtnl (View v) { // 响 应 单 击 "登录 "按钮 
String account = myEditAccount. getText( ) . toString( ) ; 
String password = myEditPassword. getText( ). toString( ) ; 
// 如 果 账 户 是 :luobin, 密码 是 :123456, 则 认为 登录 成 功 
if (account. equals("luobin") && password. equals("123456")) { 
myEditor = mySharedPreferences. edit(); // 获 取 SharedPreferences. Editor 对 象 
if (myCheckbox. isChecked()) { 
// 如 果 选 中 复 选 框 , 则 向 SharedPreferences. Editor 对 象 保 存 账 户 和 密码 信息 
myEditor. putBoolean(REMEMBER PASSWORD, true); 
myEditor. putString( ACCOUNT, account); 
myEditor. putString(PASSWORD, password); 
} else { myEditor.clear(); } 
myEditor. apply( ); // 提 交 数 据 
// 进 行 跳 转 ,并 finish 当前 activity 
//startActivity(new Intent(MainActivity. this, OtherActivity. class)); 
//finish( ); 
Toast. makeText (MainActivity.this, "登录 成 功 !", Toast. LENGTH_SHORT). show( ) ; 
} else { 
Toast. makeText(MainactivitYy.this， 
"账户 或 密码 错误 !"，Toast. LENGTH _ SHORT) . show( ) ; 
1} 


上 面 这 段 代 码 在 MyCode\ MySample319\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中, mySharedPreferences 三 PreferenceManager. 
getDefaultSharedPreferences(this) 用 于 获取 SharedPreferences 对 象 , 并 自动 使 用 当前 应 用 的 包 名 作 
为 前 级 来 命名 SharedPreferences 文件 。mySharedPreferences. getString (ACCOUNT,，"") 用 于 获取 
ACCOUNT 键 的 信息 。myEditor 二 mySharedPreferences. edit() 用 于 获取 一 个 SharedPreferences. 
Editor 对 象 。myEditor. putString (ACCOUNT，account ) 用 于 设置 ACCOUNT 键 的 信息 。 
myEditor. apply() 则 用 于 提交 (保存 ) 设 置 的 (所 有 ) 键 值 数 据 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample319 文件 夹 中 ， 


211 使 用 ListPreference 谈 写 单 选 按钮 值 


此 实例 主要 通过 使 用 ListPreference, 实 现在 存储 卡 上 保存 用 户 在 单 选 按钮 中 的 选择 结果 或 者 从 
存储 卡 上 读 取 用 户 在 单 选 按 钮 中 的 选择 结果 。 当 实例 运行 之 后 , 单 击 “ 年 度 精品 图 书 ” 列 表 选 项 , 则 将 
显示 “年 度 精 品 图 书 ” 对 话 框 ,在 该 对 话 框 中 任意 选中 一 个 单 选 按钮 ,然后 返回 ,如 “改善 既 有 代码 的 设 
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计 ” 单 选 按钮 ,如 图 211. 1 的 左 图 所 示 。 然 后 退出 此 应 用 ,再 重新 启动 此 应 用 , 则 在 弹出 的 Toast 中 显 
示 “ 刚 才 选 择 了 图 书 : 改善 既 有 代码 的 设计 ”, 如 图 211. 1 的 右 图 所 示 。 主 要 代码 如 下 : 


年 度 精品 图 书 
请 从 弹出 的 对 话 框 中 选择 年 度 精品 图 书 
年 度 精品 图 书 
〇 Java 编 程 思 想 
〇 垃圾 回收 的 算法 与 实现 
@ 改善 既 有 代码 的 设计 
〇 游戏 编程 算法 与 技巧 


刚才 选择 了 图 书 : 改善 既 有 代码 的 设 
计 


图 211.1 


public class MainActivity extends PreferenceActivity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
addPreferencesFromResource(R. xzm1. preferences); 
PreferenceManager myPreferenceManager = getPreferenceManager( ); 
// 根据 android:key 中 指定 的 名 称 (相当 于 ID) 获 取 首 选项 
ListPreference myListPreference = (ListPreference) 
myPreferenceManager. findPreference("myListPreference" ); 
Toast. makeText(MainActivity.this, "刚才 选择 了 图 书 :" + 


myListPreference. getValue( ), Toast. LENGTH SHORT). show( ); 
国 


上 F 面 这 段 代 码 在 MyCode\ MySample324\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 ，PreferenceManager myPreferenceManager = 
getPreferenceManager() 用 于 保存 用 户 的 设置 , 它 根据 包 名 和 布局 文件 创建 新 文件 来 实现 在 存储 卡 上 
保存 内 容 , 因 此 在 关闭 应 用 ,再 重新 启动 应 用 之 后 ,能 够 显示 此 前 用 户 在 对 话 框 的 操作 结果 。 
myListPreference= (ListPreference) myPreference Manager. findPreference ("myListPreference") 用 
于 在 XML 文件 中 查找 key 为 myListPreference 的 ListPreference。myListPreference. getValue() 用 
于 获取 ListPreference 值 。addPreferencesFromResource (R. xml. preferences) 表 示 MainActivity 加 
载 的 视图 是 xml 目录 下 的 preferences 文件 。preferences 文件 的 主要 内 容 如 下 : 


<?xml] version= "1.0" encoding = "UTF 一 8"?> 

< PreferenceScreen xmlns:android = "http://schemas.android. com/apk/res/android" 
android: key = "Screen list" 
android:title= "年 度 精 品 图 书 " 
android: summary = "请 从 弹出 的 对 话 框 中 选择 年 度 精品 图 书 ”> 
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<ListPreference android:key = "myListPreference" 

android:title= "年 度 精品 图 书 " 

android: summary = "请 从 弹出 的 对 话 框 中 选择 年 度 精品 图 书 " 

android:entries = "(Warray/list entries" 

android:entryValues = "(Warray/list entries value" 

android: dialogTitle = "年 度 精品 图 书 " 

android: defaultValue = " (Warray/list entries value2"></ListPreference> 
</PreferenceScreen > 


上 面 这 段 代 码 在 MyCode\MySample324\app\src\main\res\xml\preferences. xml 文件 中 。 在 这 
段 代 码 中 ,android:key 表示 唯一 标识 符 ,与 android:id 相 类 似 , 在 Java 代码 中 ,PreferenceManager 可 
以 以 其 为 参数 通过 findPreference() 方 法 获取 指定 的 preference。android:title 表示 标题 。android: 
summary 表示 选项 的 人 简单 说 明 。android: entries 表示 在 弹出 的 对 话 框 中 ,列表 项 显示 的 文本 内 容 ， 
注意 ,这 里 指定 的 是 一 个 数组 。android:entryValues 表示 与 android:entries 相对 应 的 值 。android : 
defaultValue 表示 当 对 应 值 不 存在 时 的 默认 值 。android:dialogTitle 表示 在 弹出 的 对 话 框 中 的 标题 
信息 。 上 述 三 个 数组 的 内 容 请 参考 源 代码 中 的 MyCode\MySample324\app\src\main\res\values\\ 
arrays. xml 文件 。 需 要 说 明 的 是 ,默认 情况 下 ,Android Studio 创建 的 工程 在 res 目录 下 没有 xml 目 
录 及 preferences. xml 文件 ,因此 需要 手动 添加 该 目录 和 文件 。 此 实例 的 完整 项 目 在 MyCode\\ 
MySample324 文件 夹 中 ， 


212 在 代码 中 锋 取 CheckBoxPreference 值 


此 实例 主要 通过 在 onSharedPreferenceChanged() 方 法 中 获取 CheckBoxPreference 的 key 值 , 实 
现在 Java 代码 中 判断 用 户 是 否 选中 了 CheckBoxPreference 控件 。 当 实例 运行 之 后 ,如 果 选 择 了 “是 
人 奋 开 局 定期 更 新 功能 ”CheckBoxPreference, 则 在 弹出 的 Toast 中 显示 “刚才 选择 了 定期 更 新 功能 1”， 
如 图 212. 1 的 左 图 所 示 。 如 果 未 选择 “是否 开启 定期 更 新 功能 ”CheckBoxPreference, 则 在 弹出 的 
Toast 中 显示 “刚才 取消 了 定期 更 新 功能 1”, 如 图 212. 1 的 右 图 所 示 。 注 意 : 此 实例 应 在 Android 手 
机 中 进行 测试 ,在 部 add oad Bd ati 显 不 Toast 站 


口 回 中 国生 Dv A 3 


MySample 


是 否 开启 定期 更 新 功能 
是 


刚才 选择 了 定期 更 新 功能 刚才 取消 了 定期 更 新 功能 ! 


图 212. 1 
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主要 代码 如 下 : 


public class MainActivity extends PreferenceActivity { 
SharedPreferences mySharedPreferences; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
addPreferencesFromResource(R. xml. preferences); 
mySharedPreferences = PreferenceManager.getDefaultSharedPreferences(this); 
mySharedPreferences. registerOnSharedPreferenceChangeListener( 
new SharedPreferences. OnSharedPreferenceChangeListener() { 
(@ Override 
public void onSharedPreferenceChanged( 
SharedPreferences sharedPreferences, String key) { 
if (key.equals("myUpdate" )) { 
Boolean myValue = sharedPreferences.getBoolean("myUpdate", false); 
if (myValue) { 
Toast. makeText (MainActivity. this, 
"刚才 选择 了 定期 更 新 功能 !'"，Toast.LENGTH_SHORT). show( ); 
} else { 
Toast. makeText (MainActivity. this, 
"刚才 取消 了 定期 更 新 功能 !"，Toast. LENGTH_SHORT). show( ); 
局 而 谨 让 上 
1 
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上 面 这 上段 代码 在 MyCode\ MySample323\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,mySharedPreferences 一 PreferenceManager. 
getDefaultSharedPreferences (this) 用 于 保存 PreferenceActivity 中 的 设置 ,根据 包 和 名 和 
PreferenceActivity 的 布局 文件 创建 一 个 键 名 保存 ,因此 当前 应 用 的 CheckBoxPreference 如 果 是 选中 
状态 ,在 关闭 应 用 ,再 重新 启动 应 用 之 后 ,该 CheckBoxPreference 仍然 是 选中 状态 ; 如 果 当 前 应 用 的 
CheckBoxPreference 是 未 选 状态 ,在 关闭 应 用 ,再 重新 启动 应 用 之 后 ,该 CheckBoxPreference 仍然 是 
未 选 状态 。myValue 二 sharedPreferences. getBoolean ("myUpdate"，false) 用 于 获取 key 为 
myUpdate 的 CheckBoxPreference 的 布尔 值 。addPreferencesFromResource(R. xml. preferences ) 表 
示 MainActivity 加 载 的 视图 是 xml 目录 下 的 preferences 文件 。preferences 文件 的 主要 内 容 如 下 : 


<?xml] version = "1.0" encoding = "UTF — 8"?> 
< PreferenceScreen xmlns:android = "http://schemas. android. com/apk/res/android"> 
< CheckBoxPreference android:key = "myUpdate" 
android:title = "是 否 开 启 定期 更 新 功能 " 
android: summaryOn = "是 " 
android: summary0ff =" 否 " 
android:defaultValue = "true"/> 
</PreferenceScreen> 


上 面 这 段 代 码 在 MyCode\MySample323\app\src\main\res\xml\preferences. xml 文件 中 。 此 实 


例 的 完整 项 目 在 MyCode\MySample323 文件 夹 中 。 


213 ”通过 PreferenceScreen 跳 转 到 WiFi 设置 


此 实例 主要 通过 使 用 PreferenceScreen ,实现 从 当前 应 用 跳 转 到 手机 的 WiFi 设置 界面 。 当 实例 
运行 之 后 ,将 显示 PreferenceScreen 列表 项 “WiFi 设置 >, 如 图 213. 1 的 左 图 所 示 。 单 击 列 表 项 “WiFi 


<》 Anaroia 引 本 应 有 300 。 实战 篇 


设置 ”, 则 将 跳 转 到 当前 手机 的 有 效 WiFi 列表, 如 图 213. 1 的 右 图 所 示 。 


MySample 


WiFi 设 置 
单 击 即 可 跳 转 到 手机 WiFi 设 置 


Linksys05506 


已 连接 


@PHICOMM_58 


FX_Guest 


@PHICOMM_0C 


JLLLLLLLLL 


TP-LINK_11BF 


CU_Q8Ba 


C-5-2 


ChinaMobile_3177 


图 213. 1 
主要 代码 如 下 : 


public class MainActivity extends PreferenceActivity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 


addPreferencesFromResource(R. xml. preferences ) ; 


} } 


上 F 面 这 段 代 码 在 MyCode\ MySample321 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,addPreferencesFromResource(R. xml，preferences) 表示 
MainActivity 加 载 的 视图 是 xml 目录 下 的 preferences 文件 ,而 不 是 默认 的 setContentView(R. 
layout. activity_ main) ,因为 此 处 不 再 需要 默认 的 activity_main 布局 。preferences 文件 的 主要 内 容 
如 下 : 


<?xml version= "1.0" encoding = "UTF 一 8"?> 
< PreferenceScreen xmlns:android = "http://schemas. android. com/apk/res/android"> 
< PreferenceScreen android:key = "wifi settings" 
android:title = "WiFi 设置 " 
android: summary = " 单 击 即 可 跳 转 到 手机 WiFi 设置 "> 
< intent android:action = "android. intent.action.MAIN" 
android: targetPackage = "com. android. settings" 
android:targetClass = "com. android. settings. wifi. WifiSettings" /> 
</PreferenceScreen > 
</PreferenceScreen > 


上 和 面 这 段 代 码 在 MyCode\MySample321\app\src\main\res\xml\preferences. xml 文件 中 。 此 实 
例 的 完整 项 目 在 MyCode\MySample321 文件 夹 中 ， 
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214 使 用 Intent 实现 在 Activity 之 间 传 递 小 图 像 


此 实例 主要 通过 使 用 Intent 的 getParcelableExtra() 方 法 ,实现 在 两 个 Activity 之 间 传 递 小 图 像 。 
当 实 例 运 行 之 后 ,将 在 MainActivity 的 ImageView 控件 中 显示 大 拇指 图 像 , 如 图 214. 1 的 左 图 所 示 ; 
单 击 “ 将 下 面 的 图 像 传 递 到 第 二 个 Activity” 按 钮 , 则 将 跳 转 到 SecondActivity, 此 时 SecondActivity 的 
ImageView 控件 是 一 片 空白 ; 单 击 “ 接 收 从 第 一 个 Activity 传递 的 图 像 ” 按 钮 , 则 在 ImageView 控件 
中 显示 从 MainActivity 传送 的 大 拇指 图 像 ,如 图 214. 1 的 右 图 所 示 。 


MySample MySample 


将 下 面 的 图 像 传 递 到 第 二 个 Activity 接收 从 第 一 个 Activity 传 递 的 图 像 


[此 [此 


图 214.1 
主要 代码 如 下 : 


// 响 应 单 击 "将 下 面 的 图 像 传 递 到 第 二 个 Activity" 按 钮 

public void onClickmyBtnSend(View v) {// 注 意 : 大 尺寸 图 像 无 法 通过 测试 
Bitmap myBitmap = ((BitmapDrawable) myImageMain. getDrawable( ) ) . getBitmap(); 
Intent myIntent = new Intent(MainActivity.this, SecondActivity.class); 
myIntent. putExtra(" myBitmap", myBitmap); 
startActivity(myIntent); 

} 


上 面 这 上段 代码 在 MyCode\ MySample442\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ，myBitmap 三 ((BitmapDrawable) myImageMain. 
getDrawable( )). getBitmap() 表 示 从 myImageMain 控件 中 获取 图 像 Bitmap。 mylIntent 一 new 
Intent( MainActivity. this，SecondActivity. class) 表示 从 MainActivity 跳 转 到 SecondActivity。 
myIntent. putExtra("myBitmap"，myBitmap) 表 示 在 从 MainActivity 跳 转 到 SecondActivity 时 附加 
图 像 myBitmap。SecondActivity 的 主要 代码 如 下 : 


// 响 应 单 击 "接收 从 第 一 个 Activity 传递 的 图 像 " 按 钮 
public void onClickmyBtnReceive(View v) { 
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Intent myIntent = getIntent( ); 
if(myIntent!= null){ 
Bitmap myBitmap = myIntent. getParcelableExtra("myBitmap" ); 
myImageSecond. setImageBitmap(myBitmap); 
Li 


上 面 这 段 代 码 在 MyCode\ MySample442\app\src\main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 上段 代码 中 , mylIntent = getIntent() 用 于 获取 传递 的 Intent， 
myBitmap 王 myIntent. getParcelableExtra("myBitmap") 用 于 从 myIntent 中 获取 myBitmap 代表 的 
图 像 数 据 。 此 外 , 当 新 增 了 SecondActivity 时 , 则 需要 在 AndroidManifest. xml 文件 中 注册 该 
Activity, 如 < activity android: name 一". SecondActivity "/ >。 此 实例 的 完整 项 目 在 MyCodeA\ 
MySample442 文件 夹 中 。 


215 ”使 用 Intent 在 Activity 之 间 传 递 图 像 和 文本 


此 实例 主要 通过 使 用 Intent 的 putExtra() 方 法 和 getSerializableExtra() 方 法 ,实现 在 两 个 
Activity 之 间 传 递 多 幅 小 图 像 和 文本 。 当 实例 运行 之 后 ,将 在 MainActivity 的 4 个 ImageView 控件 
中 显示 4 幅 小 图 像 , 如 图 215. 1 的 左 图 所 示 ; 单 击 “ 将 下 面 的 4 幅 图 像 发 送 到 第 二 个 Activity” 按 钮 , 则 
将 跳 转 到 SecondActivity ,此 时 SecondActivity 的 4 个 ImageView 控件 是 一 片 空 白 ; 单 击 “接收 从 第 
一 个 Activity 发 送 的 4 幅 图 像 ” 按 钮 , 则 在 4 个 ImageView 控件 中 显示 从 MainActivity 传 来 的 4 幅 小 
图 像 ,并 在 弹出 的 Toast 中 显示 同步 传递 的 文本 ,如 图 215. 1 的 右 图 所 示 。 


图 215.1 


主要 代码 如 下 : 


// 响 应 单 击 "将 下 面 的 4 幅 图 像 发 送 到 第 二 个 Activity" 按 钮 
public void onClickmyBtnSend(View v) { 
try{ 
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Bitmap myBitmapl ((BitmapDrawable) myImageViewAl. getDrawable( ) ) . getBitmap( ); 
Bitmap myBitmap2 ((BitmapDrawable) myImageViewA2. getDrawable( )).getBitmap( ); 
Bitmap myBitmap3 = ((BitmapDrawable) myImageViewA3.getDrawable()).getBitmap(); 
Bitmap myBitmap4 = ((BitmapDrawable) myImageViewA4.getDrawable()).getBitmap(); 
ArrayList < HashMap < String, Object >> myArray = new ArrayList <>(); 
HashMap < String, Object > myItem = new HashMap <>(); 
myItem. put("myImage", myBitmapl ) ; 
myItem. put ("myName"," 夺 命 三 头 小 ")，; 
myArray. add(myItem) ; 
myItem = new HashMap <>(); 
myItem. put("myImage", myBitmap2); 
myItem. put("myName", "天启"); 
myArray. add( myItem); 
myItem = new HashMap <>(); 
myItem. put("myImage", myBitmap3); 
myItem. put("myName", "变形 金刚 "); 
myArray. add(myItem) ; 
myItem = new HashMap <>( ) ; 
myItem. put("myImage", myBitmap4); 
myItem. put("myName", "速度 与 激情 "); 
myArray. add(myItem); 
Intent myIntent = new Intent(MainActivity.this,SecondActivity.class); 
myIntent. putExtra("arrayList", myArray); 
startActivity(myIntent ); 
}catch (Exception e){ 
Toast. makeText (getApplicationContext( ) ， 
e. getMessage( ) .toString(), Toast. LENGTH_ SHORT). show( ) ; 


I 


上 面 这 段 代 码 在 MyCode\ MySample463\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ，ArrayList< HashMap < String，Object >> myArray = 
new ArrayList <>() 用 于 创建 存放 键 值 对 的 数组 列表 。HashMap < String,Object > myItem 一 new 
HashMap <>() 用 于 创建 一 个 哈 硕 键 值 对 myItem。myItem. put("myImage" ，myBitmapl) 用 于 将 
myBitmapl 保存 在 喻 希 键 值 对 myItem 中 ,并 取 键 名 为 myImage。myItem. put("myName"," 夺 命 三 
头 效 ") 用 于 将 文本 “ 夺 命 三 涉 效 ”保存 在 哈 希 键 值 对 myItem 中 ,并 取 键 名 为 myName。myArray. add 
(myItem) 用 于 将 喻 希 键 值 对 myItem 添加 到 数组 列表 myArray 中 。myIntent 一 new Intent 
(MainActivity.。 this， SecondActivity. class ) 表示 myIntent 用 于 从 MainActivity 跳 转 到 
SecondActivity。myIntent. putExtra ("arrayList" ，myArray) 表 示 myIntent 将 要 传递 的 图 像 文本 数 
组 列表 myArray。SecondActivity 的 主要 代码 如 下 : 


// 响 应 单 击 "接收 从 第 一 个 Activity 发 送 的 4 幅 图 像 "按钮 
public void onClickmyBtnReceive(View v) { 
try{ 
Intent myIntent = getIntent( ); 
if(myIntent!= null)t{ 
String myNamel, myName2, myName3, myName4; 
ArrayList < HashMap < String, Object >> myArray = (ArrayList < HashMap < String, 
Object >>) myIntent. getSerializableExtra("arrayList" ); 
myImageViewB1. setImageBitmap( (Bitmap) myArray. get(0).get("myInmage" )); 
myImageViewB2. setImageBitmap( (Bitmap) myArray. get(1).get("myImage" )); 
myImageViewB3. setImageBitmap( (Bitmap) myArray. get(2).get("myImage" )); 
myImageViewB4. set ImageBitmap( (Bitmap) myArray. get(3).get("myInage" )); 
myNamel = (String) myArray. get(0).get("myName" ); 
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myName2 = ( String) myArray. get(1).get("myName" ); 
myName3 = (String) myArray. get(2).get(" myName" ); 
myName4 = (String) myArray. get(3).get(" myName" ); 
Toast. makeText (getApplicationContext(), "成 功 接收 的 电影 海报 分 别 是 :" 
+ myNamel +"" +myName2+"" + myName3 + "、 + myName4, Toast.LENGTH SHORT). show( ); 
} }catch (Exception e){ 
Toast. makeText (getApplicationContext( ), 
e. getMessage( ) . toString( ), Toast. LENGTH SHORT). show( ); 


} } 


上 面 这 段 代 码 在 MyCode\ MySample463\app\src\ main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 段 代 码 中 ,myIntent = getIntent() 用 于 获取 传递 的 Intent， 
ArrayList < HashMap < String, Object >> myArray = (ArrayList < HashMap < String, Object >>) 
myIntent. getSerializableExtra("arrayList") 用 于 从 Intent 中 获取 名 称 为 arrayList 代表 的 图 像 文本 
数组 列表 。myImageViewB1. setImageBitmap((Bitmap) myArray. get(0). get("mylImage")) 用 于 在 
myImageViewB1l 控件 中 设置 从 图 像 文 本 数组 列表 传递 的 图 像 。myNamel= (String) myArray. get 
(0). get("myName") 用 于 获取 从 图 像 文本 数组 列表 中 传递 的 文本 。 注 意 : 在 接收 Intent 传递 的 图 像 
或 文本 时 ,需要 进行 数据 类 型 强制 转换 ; 此 外 ,图 像 过 大 或 过 多 可 能 会 导致 应 用 崩溃 ,在 实际 应 用 中 需 
要 考虑 这 一 问题 。 需 要 注意 的 是 , 当 新 增 了 SecondActivity, 则 需要 在 AndroidManifest. xml 文件 中 
注册 该 Activity, 如 < activity android:name 王 ". SecondActivity"/ >。 此 实例 的 完整 项 目 在 MyCode\ 
MySample463 文件 夹 中 。 


216 ”使 用 Intent 在 Activity 之 间 传 递 集合 数据 


此 实例 主要 通过 使 用 putSerializable() 方 法 和 getSerializableExtra() 方 法 ,实现 在 两 个 Activity 
之 间 传 递 集 合 数据 。 当 实例 运行 之 后 ,在 MainActivity 的 输入 框 中 输入 两 本 书 名 ,如 图 216. 1 的 左 图 
所 示 , 单 击 “发 送 数据 ?按钮 , 则 将 跳 转 到 SecondActivity。 在 SecondActivity 中 单 击 “ 接 收 数据 ?按钮 ， 
则 在 弹出 的 Toast 中 显示 在 MainActivity 的 输入 框 中 输入 的 两 本 书 名 ,如 图 216. 1 的 右 图 所 示 。 
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MySample MySample 


发 送 数据 
第 一 本 书 名 : 垃圾 回收 的 算法 与 实现 
第 二 本 书 名 : 软件 工程 的 本 质 _ 


传递 的 集合 数据 如 下 : 
s 书 是 : 垃圾 回收 的 算法 与 实现 
一 本 书 是 : 软件 工程 的 本 质 


图 216.1 
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主要 代码 如 下 : 


public void onClickmyBtnSend(View v) { // 响 应 单 击 " 发 送 数据 "按钮 
Intent myIntent = new Intent(MainActivity.this,SecondActivity.class); 
String myBookl = myEditText1. getText( ).toString( ); 
String myBook2 = myEditText2. getText( ).toString( ); 
myList. add(myBookl ) ; 
myList. add(myBook2); 
Bundle myBundle = new Bundle( ); 
myBundle. putSerializable("myBook", (Serializable)myList); 
myIntent. putExtras(myBundle); 
startActivity(myIntent); 


上 面 这 段 代 码 在 MyCode\ MySample810\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ，myBundle. putSerializable ("myBook"，(Serializable) 
myList) 表示 将 myList 中 的 集合 数据 序列 化 之 后 存放 在 myBundle 中 。myIntent. putExtras 
(myBundle) 表示 将 myBundle 附加 在 myIntent 上 , 当 执 行 startActivity (myIntent ) 实现 从 
MainActivity 跳 转 到 SecondActivity 时 ,集合 数据 将 同时 被 传递 。SecondActivity 的 主要 代码 如 下 : 


public void onClickmyBtnReceive(View v) { // 响 应 单 击 "接收 数据 "按钮 
String myInfo = "传递 的 集合 数据 如 下 :\n"; 
Intent myIntent = getIntent( ); 
myList = (List< String>)myIntent.getSerializableExtra("myBook" ); 
myInfo+= "第 一 本 书 是 :" + myList.get(0) + "\n"; 
myInfo+= "第 二 本 书 是 :" + myList. get(1); 
Toast. makeText (getApplicationContext(), myInfo, Toast.LENGTH SHORT). show( ); 


上 面 这 段 代 码 在 MyCode\ MySample810\app\src\main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 段 人 代码 中 , myList 二 (List < String >) myIntent. 
getSerializableExtra("myBook") 表 示 从 myIntent 获取 集合 数据 ,myBook 是 数据 名 称 , 必 须 与 发 送 方 
的 名 称 一 致 。 此 外 , 当 新 增 了 SecondActivity, 则 需要 在 AndroidManifest. xml 文件 中 注册 该 


MySample810 文件 夹 中 。 


217 在 Intent 传递 数据 时 使 用 Bundle 携带 数组 


此 实例 主要 演示 了 当 使 用 Intent 在 两 个 Activity 之 间 传 递 数 据 时 ,使 用 Bundle 携带 数组 数据 。 
当 实 例 运 行 之 后 ,在 MainActivity 中 单 击 “获取 并 发 送 所 有 应 用 包 名 数据 ?按钮 ,如 图 217. 1 的 左 图 所 
示 , 则 将 跳 转 到 SecondActivity。 在 SecondActivity 中 单 击 "接收 并 显示 所 有 应 用 包 名 数据 ?按钮 , 则 
在 TextView 控件 中 显示 在 MainActivity 中 获取 的 所 有 应 用 包 名 信息 ,如 图 217. 1 的 右 图 所 示 。 
主要 代码 如 下 : 


// 响 应 单 击 "获取 并 发 送 所 有 应 用 包 名 数据 "按钮 
public void onClickmyBtnSend(View v) { 
final PackageManager packageManager = 
getApplicationContext( ). getPackageManager( ); // 获 取 所 有 应 用 包 名 数据 
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List < PackageInfo > packageInfos = packageManager.getInstalledPackages(0); 


ArrayList < String > packageNames = new ArrayList < String>(); 
if (packageInfos != null) { 
for (int i = 0; i< packageInfos.size(); i++) { 
String myPackageName = packageInfos. get(i). packageName; 
packageNames. add( myPackageName); 
bE 
// 发 送 所 有 应 用 包 名 数据 
Intent myIntent = new Intent(MainActivity.this,SecondActivity.class); 
Bundle myBundle = new Bundle( ); 
myBundle. putStringArrayList("myNames" , packageNames); 
myIntent. putExtras(myBundle); 
startActivity(myIntent ); 
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MySample MySample 


获取 并 发 送 所 有 应 用 包 名 数据 接收 并 显示 所 有 应 用 包 名 数据 
当前 手机 安装 的 应 用 包 名 如 下 : 


com.mediatek.ppl 
com,.mediatek.apkinstaller 
com.Widget.polaroid 
com.android.providers.telephony 
com.adups.fota.sysoper 
com.android.providers.calendar 
com.android.providers.media 
com.qiyi.video 
com.mediatek.fwk.plugin 
com.mobiletools.systemhelper 
com.android.wallpapercropper 
com.mediatek.schpwronoff 
com,mediatek.videoplayer 
com.tencent.qqmusic 
com.android.documentsui 
com.android.galaxy4 
com.android.externalstorage 
com.mediatek.ygps 
com.android.htmlviewer 
com.tencent.androidqqmail 
com.android.quicksearchbox 
com.android.mms.service 
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上 面 这 段 代码 在 MyCode\ MySample446 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ，Bundle 是 一 个 简单 的 数据 携带 包 , 当 使 用 Bundle 携带 字 
符 串 类 型 的 数组 数据 时 ,应 使 用 Bundle 的 putStringArrayList() 方 法 。putStringArrayList() 方 法 的 
语法 格式 如 下 : 


putStringArrayList((@Nullable String key, (@Nullable ArrayList < String> value) 


其 中 ,参数 String key 表示 数据 名 称 。 参 数 ArrayList < String > value 表示 数组 类 型 的 字符 串 数 
组 。 当 将 字符 串 数组 通过 Bundle 的 putStringArrayList() 方 法 附加 到 Bundle 对 和 象 之 后 ,还 需要 将 
Bundle 对 象 通 过 Intent 的 putExtras() 方 法 附加 到 Intent。 当 使 用 startActivity(myIntent) 方 法 从 
MainActivity 跳 转 到 SecondActivity 时 ,myIntent 包含 的 Bundle 数组 就 传递 到 了 SecondActivity。 
SecondActivity 的 主要 代码 如 下 : 
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// 响 应 单 击 " 接 收 并 显示 所 有 应 用 包 名 数据 "按钮 
public void onClickmyBtnReceive(View v) { 
String myText = "当前 手机 安装 的 应 用 包 名 如 下 :"; 
Intent myIntent = getIntent( ); 
Bundle myBundle = myIntent. getExtras( ); 
ArrayList < String > packageNames = myBundle.getStringArrayList("myNames" ); 
if (packageNames != null) { 
for (int i = 0; i< packageNames. size(); i++) { 
String myName = packageNames. get(i).toString(); 
myText += "\n" + myName; 
国 
myTextView. SetText(myText ); 


上 面 这 上段 代码 在 MyCode\ MySample446 \app\src\ main\java\com\bin\luo\mysample\ 
SecondActivity. java 文件 中 。 在 这 段 代 码 中 ,myBundle = 二 myIntent. getExtras() 用 于 从 Intent 取得 
Bundle 对 象 。packageNames 一 myBundle. getStringArrayList ("myNames") 用 于 从 Bundle 对 象 中 
获取 数据 名 称 为 myNames 的 字符 串 数 组 。 需要 说 明 的 是 , putStringArrayList ( ) 和 
getStringArrayList() 方 法 仅 支 持 字符 串 类 型 的 数组 数据 传递 ,其 他 类 型 的 数组 数据 传递 应 使 用 
Bundle 提供 的 其 他 方法 , 如 传递 int 类 型 的 数组 数据 应 使 用 putIntegerArrayList ( )、 
getIntegerArrayList() 等 。 此 外 , 当 新 增 了 SecondActivity, 则 需要 在 AndroidManifest. xml 文件 中 注 
册 该 Activity, 如 < activity android: name 一 ". SecondActivity"/ >。 此 实例 的 完整 项 目 在 MyCode\ 
MySample446 文件 夹 中 。 


218 ”使 用 Intent 在 Service 和 Activity 之 间 传 递 数 据 


此 实例 主要 通过 在 自 定 义 Service 中 采集 时 间 并 发 送 广 播 ,实现 使 用 Intent 以 广播 的 方式 在 
Service 和 Activity 之 间 传 递 数据 。 当 实例 运行 之 后 , 单 击 “启动 服 务 ? 按 钮 , 则 上 自 定 义 Service 采集 的 
时 间 数 据 将 以 广播 的 形式 发 送 ,并 被 IntentFilter 过 滤 接 收 , 即 逐 秒 显示 当前 时 间 , 如 图 218. 1 的 左 图 
所 示 ; 单 击 “ 停 止 服务 ”按钮 , 则 服务 停止 ,显示 的 当前 时 间 不 再 更 新 ,如 图 218. 1 的 右 图 所 示 。 
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MySample MySample 


启动 服务 停止 服务 启动 服务 停止 服务 
当前 时 间 : 2018-06-20 18:26:06 当前 时 间 : 2018-06-20 18:26:17 


图 218.1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

TextView myTextView; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
myTextView = (TextView) findViewById(R. id.myTextView); 
IntentFilter myIntentFilter = new IntentFilter(); 
myIntentFilter.addAction("TimeService" ); // 设 置 过 滤 Action 字符 串 
registerReceiver(new ServiceReceiver( )，myIntentFilter);  // 动 态 注册 广播 

} 


public void onClickBtnl (View v) { // 响 应 单 击 "启动 服务 "按钮 
Intent myIntent = new Intent(MainActivity.this, TimeService.class); 
startService(myIntent); // 通 过 Intent 启动 服务 


Toast. makeText (MainActivity. this, 
"已 开启 时 钟 服务 !"，, Toast.LENGTH SHORT). show(); 
} 


public void onClickBtn2(View v) { // 响 应 单 击 "停止 服务 "按钮 
Intent myIntent = new Intent(MainActivity.this, TimeService.class); 
stopService(myIntent); // 通 过 Intent 停止 服务 


Toast. makeText (MainActivity. this, 
"已 停 止 时 钟 服务 !"， Toast.LENGTH SHORT). show(); 


} 
// 接 收 从 服务 发 送 数 据 的 广播 
public class ServiceReceiver extends BroadcastReceiver { 
@Override 
public void onReceive(Context context, Intent intent) { 
myTextView. setText(" 当前 时 间 :”+ 
intent. getStringExtra("myTime" ) ); // 通 过 TextView 控件 显示 当前 时 间 
二)} 


上 F 面 这 段 代 码 在 MyCode\ MySample911\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,Intent myIntent 一 new Intent(MainActivity。 this， 
TimeService. class) 用 于 根据 指定 的 MainActivity 和 TimeService 创建 Intent,TimeService 是 获取 当 
前 时 间 的 自 定义 Service,TimeService 类 的 主要 代码 如 下 : 


public class TimeService extends Servicef{ 
Timer myTimer; TimerTask myTimerTask; 
(Override 
public void onCreate( ){ 
super. onCreate( ) ; 
myTimer = new Timer( ); 
myTimerTask = new TimerTask(){ 
(Override 
public void run( ){ 
Intent myIntent = new Intent(); 
SimpleDateFormat mySimpleDateFormat = 
new SimpleDateFormat("yyyy - MM - dd HH: mm: ss" ); // 格 式 化 日 期 
myIntent. putExtra("myTime" ,mySimpleDateFormat. format (new Date( ) ) ) ; 
myIntent. setAction(" TimeService" ); 
sendBroadcast (myIntent ); // 通 过 广播 传递 指定 数据 
ps 
myTimer. schedule(myTimerTask, 0,1000); // 每 隔 1 秒 发 送 一 次 广播 
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(Override 
public IBinder onBind( Intent intent){return nul1;} 
(Override 
public void onDestroy( ){ 
super. onDestroy!( ) ; 
myTimerTask. cancel( ) ; 
myTimerTask = null; 
myTimer. cancel( ) ; 


时 


// 停 止 并 销毁 定时 器 及 定时 任务 对 象 


上 面 这 段 代 码 在 MyCode\ MySample911\app\src\ main\java\com\bin\luo\mysample\ 
TimeService. java 文件 中 。 一 般 情 况 下 , 当 在 项 目 中 添加 Service 之 后 ,通常 应 该 在 AndroidManifest. 
xml 文件 中 进行 注册 ,主要 代码 如 下 : 

< service android:name = ".TimeService" 


android: enabled = "true" 
android: exported = "true"></service> 


上 和 面 这 段 代码 在 MyCode\MySample911\app\src\main\AndroidManifest. xml 文件 中 。 此 实例 
的 完整 项 目 在 MyCode\MySample911 文件 夹 中 ，。 


219 使 用 FileInputstream 和 FileOutputStream 谈 取 和 保存 
文本 文件 


此 实例 主要 通过 使 用 FileInputStream 和 FileOutputStream ,实现 在 应 用 中 读 取 和 保存 文本 文件 。 当 
实例 运行 之 后 ,在 “文件 名 称 : ”输入 框 中 输入 文件 名 称 , 在 “文件 内 容 : ”输入 框 中 输入 内 容 , 单 击 “ 保 存 文 
本 文件 ”按钮 , 则 将 根据 指定 的 文件 名 称 保存 文本 内 容 到 “data\data\ 应 用 包 名 \files” 中 。 在 “文件 名 称 : ” 
输入 框 中 输入 文件 名 称 , 单 击 “ 读 取 文 本 文件 ”按钮 , 则 将 从 “data\data\ 应 用 包 名 \files” 中 读 取 指定 文件 的 
文本 内 容 到 “文件 内 容 : ”输入 框 中 ,如 图 219. 1 的 左 图 (修改 前 ) 和 右 图 (修改 后 ) 所 示 。 
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MySample MySample 


保存 文本 文件 读 取 文本 文件 保存 文本 文件 读 取 文 本 文件 


件 名 称 : mydata .txt 件 名 称 : mydata txt 


件 内 容 : 联合国 是 第 二 次 世界 大 战 后 成 立 的 
国际 组 织 ， 是 一 个 由 主权 国家 组 成 
的 国际 组 织 。1945 年 10 月 24 日 ， 


文件 内 容 : 联合 国 是 第 二 次 世界 大 战 后 成 立 的 
国际 组 织 ， 是 一 个 由 主权 国家 组 成 
的 国际 组 织 。1945 年 10 月 24 日 ， 


在 美国 旧金山 签订 生效 的 《联合 国 
宪章 》， 标 志 着 联合 国正 式 成 立 。 
联合 国 致力 于 促进 各 国 在 国际 法 、 
国际 安全 、 经 济 发 展 、 社 会 进步 、 


人 权 及 实现 世界 和 平方 面 的 合作 。 


图 


在 美国 旧金山 签订 生效 的 《联合 国 
宪章 》 ， 标 志 着 联合 国正 式 成 立 。 
联合 国 致力 于 促进 各 国 在 国际 法 、 
国际 安全 、 竺 会 进步 、 人 权 及 实现 
世界 和 平方 面 的 合作 。 
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public void onClickmyBtnl(View v) { // 响 应 单 击 " 保 存 文本 文件 "按钮 
File0utputStream myOutStream = null; 
BufferedWriter myBufferedWriter = null; 
try { 
myOutStream = openFileOQutput(myEditName.getText().toString(), 

Context. MODE PRIVATE); 
myBufferedWriter = new BufferedWriter(new OutputStreamWriter(myOutStream) ) ; 
myBufferedWriter.write(myEditText. getText().toString( )); 

} catch (IOException e) { e.PprintStackTrace( );} 
finally { 
try { 
if (myBufferedWriter != null) { myBufferedWriter.close(); } 
} catch (IOException e) { e.printStackTrace( ); } 
} 
Toast. makeText (getApplicationContext( ), 
"已 经 成 功 保存 文件 "，Toast. LENGTH_SHORT) .show( ) ; 
} 
public void onClickmyBtn2(View v) { // 响 应 单 击 " 读 取 文 本 文件 "按钮 
FileInputStream myInputStream = null; 
BufferedReader myBufferedReader = null; 
StringBuilder myText = new StringBuilder(); 
tryl 
myInputStream = openFileInput(myEditName.getText().toString()); 
myBufferedReader = new BufferedReader(new InputStreamReader(myInputStreanm)); 
String myLines = ""; 
while ((myLines = myBufferedReader.readLine())!= null){ 
myText. append(myLines); 
} } catch (IOException e) { e.printStackTrace( );} 
finally { 
if (myBufferedReader != null) { 
try { 
myBufferedReader. close( ); 
} catch (IOException e) { e.printStackTrace( ); } 
这 
myEditText. setText(myText. toString( ) ); 
} 


上 面 这 段 代 码 在 MyCode\ MySample420\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myOutStream 三 openFileOutput(myEditName. getText 
(). toString() ，Context. MODE_PRIVATE) 用 于 获取 文件 输出 流 , 第 一 个 参数 是 文件 名 称 , 第 二 个 
参数 指定 文件 的 操作 模式 ; 文件 的 操作 模式 有 : MODE_PRIVATE( 默 认 操 作 模 式 , 表 示 当 指定 同名 
文件 的 时 候 , 所 写 入 的 内 容 将 覆盖 原文 件 的 内 容 )、MODE_APPEND( 表 示 如 果 该 文件 已 存在 ,就 往 文 
件 里 追加 内 容 ,不 存在 则 创建 新 文件 )。 此 实例 的 完整 项 目 在 MyCode\MySample420 文件 夹 中 。 


220 ”将 浮雕 风格 的 特效 文字 保存 为 图 像 文件 
此 实例 主要 通过 使 用 TextView 的 getDrawingCache() 方 法 和 Bitmap 的 compress() 方 法 ,实现 


将 特效 文字 保存 为 图 像 文件 。 当 实例 运行 之 后 ,将 在 TextView 控件 中 显示 浮雕 文字 “ 炫 酷 ”, 单 击 “ 将 
文字 保存 为 图 像 ” 按 钮 ,将 把 浮雕 效果 的 “ 炫 酷 ”二 字 保 存在 SD 卡 的 Pictures 目录 中 ,效果 分 别 如 


图 220. 1 的 左 图 和 右 图 所 示 。 


MySample 


将 文字 保存 为 图 像 


翁 抑 


void OnClickButtonl (View view) { 
myTextView. setDrawingCacheEnabled( true); 
myTextView. buildDrawingCache( ); 
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人 Screenshots 
Ea 1516951597484jpg 
大 小 43.69 KB 


加 1516951966354jpg 
大 小 43.69 KB 


// 响 应 单 击 "将 文字 保存 为 图 像 "按钮 


Bitmap myBitmap = myTextView. getDrawingCache( ); 


try { 


Uri myImageUri = getContentResolver(). insert( 


MediaStore. Images. Media. EXTERNAL CONTENT URI, new ContentValues( )); 
OutputStream myOutStream = getContentResolver().openOutputStream(myImageUri); 
myBitmap. compress(Bitmap. CompressFormat. JPEG, 90, myOutStreanm); 


Toast. makeText (MainActivity. this, 


"将 文字 保存 为 图 像 操 作成 功 !"，Toast. LENGTH_SHORT). show( ); 


} catch (Exception e) { e. printStackTrace(); } 


} 


文件 和 数据 


上 面 这 上段 代码 在 MyCode\ MySample682\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myBitmap 二 myTextView. getDrawingCache() 用 于 将 
TextView 控件 绘制 的 内 容 ( 浮 雕 文字) 转换 成 Bitmap。 myBitmap. compress (Bitmap. 
CompressFormat. JPEG,， 90, myOutStream) 用 于 将 Bitmap 的 内 容 保存 为 图 像 文 件 。 此 外 ,访问 SD 
卡 需 要 在 AndroidManifest. xml 文件 中 添加 < uses-permission androlid:name 一 “android. permission. 


WRITE_ EXTERNAL STORAGE"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample682 文件 夹 中 。 


221 在 SD 卡 上 将 Bitmap 保存 为 PNG 图 像 文件 


此 实例 主要 通过 使 用 Bitmap 类 的 compress() 方 法 ,实现 以 文件 流 方 式 将 Bitmap 图 像 在 SD 卡 上 
保存 为 PNG 格式 的 图 像 文 件 。 当 实例 运行 之 后 ,在 "图像 文件 : ”输入 框 中 输入 图 像 文 件 的 保存 路 
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径 , 如 图 221. 1 的 左 图 所 示 , 单 击 “ 保 存 图 像 ” 按 钮 ,将 把 下 面 显示 的 图 像 在 SD 卡 上 保存 为 PNG 格式 
的 图 像 文 件 , 如 图 221. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 
public void onClickmyBtnl (View v) { // 响 应 单 击 " 保 存 图 像 " 按 钮 
try { 
FileOutputStream myOutStream = 


new File0utputStream(myEditText. getText( ) .toString() ); 
Bitmap myBitmap = ((BitmapDrawable)myImageView. getDrawable( ) ) . getBitmap(); 
myBitmap. compress(Bitmap. CompressFormat. PNG, 100, myOutStreanm); 
myOutStream. flush( ); 
myOutStream. close( ); 
Toast. makeText(this, "保存 成 功 ! " ,Toast.LENGTH SHORT). show( ); 
} catch (Exception e) { 
Toast. makeText(this, e. getMessage( ), Toast. LENGTH SHORT) . show( ) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySample222\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myOutStream 一 new FileOutputStream (myEditText. 
getText(). toString()) 用 于 根据 指定 的 文件 路 径 创 建文 件 输出 流 。myBitmap 二 (BitmapDrawable) 
myImageView。getDrawable ( )). getBitmap ( ) 用 于 获取 ImageView 控件 的 图 像 。myBitmap. 
compress( Bitmap. CompressFormat. PNG，100，myOutSstream ) 用 于 根据 文件 输出 流 和 Bitmap. 
CompressFormat. PNG 参数 将 图 像 保 存 为 PNG 格式 文件 。 此 外 ,在 SD 卡 写 入 文件 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 二 "android. permission. WRITE 
EXTERNAL_STORAGE"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample222 文件 夹 中 。 
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222 ”从 手机 相册 中 选择 图 像 文件 并 裁 王 头像 


此 实例 主要 通过 使 用 startActivityForResult() 方 法 调用 相册 和 裁剪 Activity ,实现 从 手机 相册 中 
选择 图 像 文件 并 裁剪 头像 。 当 实例 运行 之 后 , 单 击 “ 从 图 库 中 选择 图 像 文件 ”按钮 , 则 将 显示 相册 
Activity, 如 图 222. 1 的 左 图 所 示 。 在 相册 Activity 中 选择 一 个 图 像 文 件 , 则 将 显示 裁剪 Activity; 在 
裁剪 Activity 中 移动 网 格 线 剪裁 图 像 ,如 图 222. 1 的 右 图 所 示 , 单 击 左 上 角 的 “保存 ”按钮 , 则 在 主 
Activity 中 显示 裁剪 的 头像 。 
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图 222.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
/* 请 求 识别 码 * / 
private static final int CODE GALLERY REQUEST = 0xa0; 
private static final int CODE RESULT REQUEST = 0xa2; 
// 裁 前 后 图 像 的 宽 X 和 高 Y, 480 x* 480 的 正方 形 . 
private static int output X = 480; 
private static int output Y = 480; 
private ImageView myImageView = null; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
} 
public void onClickmyBtnl (View v) { // 响 应 单 击 "从 图 库 中 选择 图 像 文 件 "按钮 
Intent myIntent = new Intent(); 
myIntent. setType(" image/ * " ); 
myIntent. setAction( Intent. ACTION GET CONTENT ) ; 
startActivityForResult(myIntent, CODE GALLERY REQUEST); 
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} 
(Override 


protected void onRctivityResult( int requestCode, int resultCode, Intent intent) { 
if (resultCode == RESULT CANCELED) { // 如 果 用 户 没 有 进行 有 效 操作 , 则 直接 返回 
Toast. makeText (getApplication( )," 取消 "，, Toast. LENGTH LONG) . show( ); 
return; 
} 
switch (requestCode) { 
case CODE GALLERY REQUEST: 
cropRawPhoto( intent. getData( ) ); 
break; 
Case CODE RESULT REQUEST: 
if (intent != null) { setImageToHeadView(intent); } 


break; 
} 
super. onActivityResult(requestCode, resultCode, intent); 
} 
public void cropRawPhoto(Uri uri) { // 裁 剪 原 始 图 像 


Intent intent = new Intent("com. android. camera. action. CROP"”) ; 
intent. setDataAndType(uri, "image/ * "); 
intent. putExtra("crop", "true" ); 
intent. putExtra("aspectX", 1); 
intent. putExtra("aspectY", 1); 
intent. putExtra("outputX", output X); // 裁 剪 图 像 的 宽 和 高 
intent. putExtra("outputY", output Y); 
intent. putExtra(" return -~ data", true); 
startActivityForResult(intent, CODE RESULT REQUEST); 
} 
private void setImageToHeadView( Intent intent){ // 保 存 裁剪 图 像 , 并 设置 头像 
Bundle extras = intent.getExtras( ); 
if (extras != null) { 
Bitmap myBmp = extras.getParcelable("data" ); 
myImageView. setImageBitmap(myBmp); 
}}} 


上 面 这 上段 代码 在 MyCode\ MySamplel86 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,startActivityForResult(mylIntent, CODE_GALLERY_ 
REQUEST) 用 于 根据 myIntent 请 求 码 访问 相册 ,并 在 onActivityResult (int requestCode，int 
resultCode ,Intent intent) 方 法 中 处 理 访问 结果 。 由 于 每 个 Activity 都 可 以 启动 任意 的 子 Activity 并 
等 待 结果 , 而 结果 处 理 方 法 只 有 一 个 onActivityResult (int requestCode, int resultCode, Intent 
intent) ,因此 为 了 区 别 请 求 的 Activity 是 谁 ,Android 将 每 个 请 求 设 定 为 一 个 大 于 等 于 0 的 值 ,这 就 是 
requestCode。 由 此 ,在 onActivityResult() 方 法 中 即 可 利用 requestCode 区 别 不 同 的 Activity 并 返回 
结果 。 在 此 实例 中 , requestCode 主要 有 CODE GALLERY REQUEST 和 CODE RESULT _ 
REQUEST, 它 表明 onActivityResult() 至 少 执行 了 两 次 ,一 次 是 获取 选择 的 图 像 文件 ,第 二 次 是 获取 
裁剪 之 后 的 头像 。 此 实例 的 完整 项 目 在 MyCode\MySamplel186 文件 夹 中 ， 


223 在 ListView 上 加 载 手机 外 存 的 图 像 文件 


此 实例 主要 实现 了 在 ListView 上 加 载 手 机 外 部 存储 介质 上 的 所 有 图 像 文件 。 当 实例 运行 之 后 ， 
将 在 ListView 控件 上 显示 手机 外 部 存储 上 的 所 有 图 像 文件 的 名 称 和 大 小 ,如 图 223. 1 的 左 图 所 示 ; 
单 击 在 ListView 控件 上 的 任意 图 像 文件 名 称 ( 列 表 项 ) , 则 在 弹出 的 对 话 框 中 显示 该 图 像 ,如 图 223. 1 
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的 右 图 所 示 。 
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图 223. 1 


public class MainActivity extends AppCompatActivity { 
private ListView myListView; 
(Override 
protected void onCreate( @ Nullable Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity_main); 
myListView = (ListView) findViewById(R. id. myListView) ; 
GetImageFiles( ) ; 
} 
ArrayList myFiles = new ArrayList(); 
ArrayList mySizes = new ArrayList(); 
ArrayList < String > myPathData = new ArrayList(); 
ArrayList < HashMap < String, Object >> myItems = new ArrayList <>(); 
private void GetImageFiles() { 
myFiles. clear( ); 
mySizes. clear( ) ; 
// 通 过 ContentResolver 查询 外 部 存储 介质 上 的 所 有 图 像 文件 信息 
Cursor myCursor = getContentResolver( ). query(MediaStore. Images. Media. EXTERNAL CONTENT URI, null, 
null, null, null); 
while (myCursor. moveToNext()) { 
String myName = myCursor. getString(myCursor. getColumnIndex( 


MediaStore. Images. Media. DISPLAY NAME)); // 显 示 名 称 
String mySize = myCursor.getString(myCursor.getColumnIndex( 
MediaStore. Images. Media. SIZE) ); // 显 示 文 件 大 小 
byte[ ] data = myCursor.getBlob(myCursor.getColumnIndex( 
MediaStore. Images. Media. DATA) ) ; // 获 取 图 像 的 位 置 数据 
myFiles.add(myName); 


mySizes.add(mySize); 
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String myPath = new String(data, 0, 
data. length - 1); // 将 位 置 数 据 转换 成 字符 串 形式 的 路 径 
myPathData. add( myPath); 
} 
for (int i = 0; i< myFiles. size(); i++) { 
HashMap < String, Object > myListItem = new HashMap <>(); 
myListItem. put("name", myFiles. get(i)); 
myListItem. put("size", mySizes. get(i)); 
myItems. add(myListItem) ; 
} 
SimpleAdapter myAdapter = new SimpleAdapter(MainActivity.this,myItems, 
R. layout. myitem, new String[ ]{" name”", "size”}, 
new int[ ]{R. id.myName, R. id.mySize} ); 
myListView. setAdapter(myAdapter ); 
// 单 击 图 像 文件 名 称 ( 列 表 项 ) 显 示 图 像 
myListView. setOnItemClickListener(new AdapterView. OnItemClickListener() { 
(Override 
public void onItemClick(AdapterView <?> parent, View view, int position, long id){ 
View myDialog = getLayoutInflater(). inflate(R. layout.mydlg, null]l); 
ImageView myImageView = (ImageView) myDialog.findViewById(R. id.myImageView); 
// 获 取 对 应 列表 项 的 指定 图 像 
Bitmap myBitmap = BitmapFactory.decodeFile(myPathData.get(position)); 
myImageView. setImageBitmap(myBitmap); 
new AlertDialog. Builder(MainActivity.this). setView(myDialog) 
. setPositiveButton(" 确定"，nul1). show(); 
二 


上 面 这 上段 代码 在 MyCode\ MySample668\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 , SimpleAdapter myAdapter 一 new SimpleAdapter 
(MainActivity. this, myltems, R. layout. myitem,new Stringl |{"name", "size"},new int| | {R. id. 
myName，R. id. mySize}) ) 表 示 根 据 myitem 布局 的 样式 创建 ListView 的 列表 项 ,关于 myitem 布局 的 
详细 内 容 请 参考 源 代 码 中 的 MyCode\MySample668\app\src\main\res\ layout\myitem. xml 文件 。 
myDialog 王 getLayoutInflater(). inflate (R. layout. mydlg，null) 表 示 加 载 mydlg 布局 作为 显示 图 像 
的 对 话 框 。 关 于 mydlg 布局 的 详细 内 容 请 参考 源 代 码 中 的 MyCode\MySample668\app\src\main\ 
res\ layout \ mydlg. xml 文件 。myCursor 一 getContentResolver ( ). query ( MediaStore. Images. 
Media. EXTERNAL _ CONTENT URI, null, null, null, null) 表 示 查 找 外 部 存储 介质 上 的 所 有 图 像 
文件 。 此 外 , 当 访 问 外 部 存储 介质 时 ,需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission 
android:name 二 "android. permission. READ _EXTERNAL_STORAGE"/> 权 限 。 需 要 说 明 的 是 ,使 
用 此 实例 的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 25. 0. 1' 依 赖 项 。 此 
实例 的 完整 项 目 在 MyCode\MySample668 文件 夹 中 ， 


224 使 用 DownloadManager 下 载 网 络 文件 


此 实例 主要 实现 了 使 用 Android 内 置 的 DownloadManager 下 载 网 络 文件 。 当 实例 运行 之 后 ,在 
“下 载 网 址 : ”输入 杠 中 输入 网 络 地 址 , 如 “http://gdown. baidu. com/data/ wisegame/ 
0904344dee4a2d92/QQ_718. apk”, 如 图 224. 1 的 左 图 所 示 ; 然后 单 击 “ 下 载 ” 按 钮 , 则 将 执行 下 载 操 
作 , 下 载 进 度 将 显示 在 通知 栏 上 ,如 图 224. 1 的 右 图 所 示 。 
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图 224. 1 
主要 代码 如 下 : 


public void onClickButtonl(View v){ // 响 应 单 击 " 下 载 "按钮 
DownloadManager myDownloadManager = 
(DownloadManager )getSystemService(DOWNLOAD SERVICE); // 获 取 下 载 管理 器 
String myUrl = myEditText. getText().toString( ); 


DownloadManager. Request myRequest = 
new DownloadManager. Request (Uri. parse(myUr])); // 向 Url 发 送 下 载 请 求 
myRequest. setDestinationInExternalPublicDir("","Q0Q. apk" ); // 设 置 下 载 保存 位 置 
myRequest. setNotificationVisibility(DownloadManager. Request. 
VISIBILITY VISIBLE NOTIFY COMPLETED); // 在 通知 栏 显示 下 载 进 度 
myDownloadManager. enqueue( myRequest ); // 将 请 求 加 入 下 载 队列 


上 面 这 段 代 码 在 MyCode\MySample892\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 上段 代码 中, myRequest. setNotificationVisibility ( DownloadManager. Request. 
VISIBILITY_VISIBLE_NOTIFY_COMPLETED) 用 于 在 通知 栏 上 显示 下 载 进度 ,此 参数 可 以 使 得 下 载 进 
度 在 下 载 过 程 中 和 下 载 完 成 后 均 可 见 。 如 果 是 myRequest. setNotificationVisibility (DownloadManager. 
Request. VISIBILITY_VISIBLE), 则 在 下 载 过 程 中 在 通知 栏 上 显示 下 载 进 度 , 下 载 完 成 之 后 自动 消 
失 。DownloadManager 是 专用 于 处 理 耗 时 较 长 的 HTTP 文件 下 载 的 系统 服务 , 它 在 后 台 进 行 下 载 ， 
并 自动 处 理 网 络 连 接 变 化 ,失败 重 试 。DownloadManager 可 以 在 自己 的 应 用 中 提交 下 载 请 求 , 在 通知 
栏 中 自动 显示 下 载 进 度 , 可 以 指定 下 载 文件 的 保存 位 置 ,并 实时 获取 下 载 进度 ,监听 下 载 结果 。 
DownloadManager 中 有 两 个 重要 的 内 部 类 。 

(1) DownloadManager. Request, 该 类 封装 一 个 下 载 请 求 添加 到 系统 下 载 希 队列 。 
(2) DownloadManager. Query ,该 类 查询 下 载 任务 ,可 实时 获取 下 载 进 度 ,下载 结 果 。 
此 外 ,访问 网 络 和 读 写 SD 卡 上 的 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 


"android. permission. INTERNET"/> 和 < uses-permission androld : 


name 一 "android. permission。WRITE EXTERNAL _ STORAGE"/ > 权限 。 此 实例 的 完整 项 目 在 


permission androld: name = 
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MyCode\MySample892 文件 夹 中 。 


225 使 用 RandomAccessFile 实现 斯 点 续 传 下 载 


此 实例 主要 通过 使 用 RandomAccessFile 的 seek() write() 等 方法 ,实现 在 存储 卡 上 写 人 下 载 文 
件 , 并 在 下 载 网 络 文件 的 过 程 中 支持 断 点 续 传 功能 。 当 实例 运行 之 后 ,在 “下 载 地 址 : ”输入 框 中 输入 
网 络 地 址 ,如 “http://gdown. baidu. com/data/wisegame/ 0904344dee4a2d92/QQ _718. apk”, 然 后 单 
击 “ 开 始 下 载 ? 按 钮 ,执行 下 载 操作 ,进度 条 同时 显示 下 载 进 度 ; 单 击 “ 暂 停 下 载 ” 按 钮 , 则 暂停 下 载 操 
作 , 同 时 暂停 进度 条 下 载 进度 ,如 图 225. 1 的 左 图 所 示 ; 再 次 单 击 “开始 下 载 ? 按 钮 , 则 在 上 次 暂停 的 位 
置 继续 执行 下 载 操 作 ,直到 下 载 完成 ; 由 于 实例 下 载 的 是 应 用 安装 包 , 因 此 在 下 载 完 成 之 后 将 自动 执 


行 安装 操作 ,如 图 225. 1 的 右 图 所 示 。 
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图 225.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
ProgressBar myProgressBar; 
EditText myEditText; 
int myPausedPosition = 0; 
boolean isPaused = false; 
Handler myHandler = new Handler() { 
(Override 
public void handleMessage(Message msg) { 
if (msg.what == 0) { 
int myFileSize = msg.argl; 
downloadFile(myFileSize); // 开 始 异 步 下 载 文 件 
} else if (msg.what == 1) { 
myProgressBar. setProgress(msg. argl ); // 实 时 显示 当前 下 载 进度 
} else if (msg.what == 2) { 
Toast. makeText (MainActivity. this, 


"下 载 完成 ! 即 将 开始 安装 ..."，Toast.LENGTH SHORT). show( ); 
Intent myAPKIntent = new Intent(Intent.ACTION VIEW); 
myAPKIntent. setDataAndType(Uri.fromFile(new File( 
Environment. getExternalStorageDirectory().getAbsolutePath( ) 
+ "/QQ.apk" )), "application/vnd.android. package - archive" ); 
startActivity(myAPKIntent ); // 下 载 完成 后 跳 转 至 APK 安装 界面 
} else if (msg.what == 3) { 
Toast. makeText(MainRctivity.this, "开始 下 载 !" ，Toast. LENGTH SHORT) . show( ); 
} } )}; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myProgressBar = (ProgressBar) findViewById(R. id.myProgressBar); 
myEditText = (EditText) findViewById(R. id.myEditText); 
} 


public void onClickButtonl (View v) { // 响 应 单 击 " 开 始 下 载 " 按 钮 
if (isPaused) isPaused = false; 
CalcFileSizel( ); 

} 

public void onClickButton2(View v) { // 响 应 单 击 "暂停 下 载 " 按 钮 


isPaused = true; 
Toast. makeText (this, "已 经 暂停 下 载 !"， Toast. LENGTH SHORT). show( ) ; 
} 


public void calcFileSize() { // 计 算 下 载 文 件 大 小 
new Thread(new Runnable() { 
@ Override 
public void run() { 
try { 


URL myUrl = new URL(myEditText.getText().toString()); 
HttpURLConnection myConnection = (HttpURLConnection) myUr]l. openConnection( ); 
myConnection. setRequestMethod("GET" ); 
myConnection. setConnectTimeout(5000); 
if (myConnection.getResponseCode() == 200) { 
Message myFileSizeMessage = new Message( ); 
myFileSizeMessage.what = 0; 
myFileSizeMessage.argl = myConnection. getContentLength( ); 
// 获 取 文 件 大 小 ,并 发 送 至 Handler 
myHandler. sendMessage(myFileSizeMessage); 
} } catch (Exception e) { 
Log.e("mytag", e.toString()); 
} } }). start(); 
} 
public void downloadFile(final int fileSize) { 
new Thread(new Runnable() { 
@ Override 
public void run() { 
try { 
URL myUrl = new URL(myEditText.getText().toString()); 
HttpURLConnection myConnection = 
(HttpURLConnection) myUr1. openConnection( ); 
myConnection. setRequestMethod(" GET" ); 
myConnection. setConnectTimeout(5000); 


文件 和 数据 
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myConnection. setRequestProperty(" Range", "bytes=" 
+ myPausedPosition + "-"” + fileSize);  // 指 定 下载 的 区 间 范 围 
if (myConnection. getResponseCode() == 206) { 
myHandler. sendEmptyMessage( 3); 
InputStream myInputStream = myConnection.getInputStream( ); 
File myFile = new File(Environment.getExternalStorageDirectory(). 


getAbsolutePath() + "/00.apk" ); // 设 置 下 载 之 后 的 保存 位 置 
RandomAccessFile myRandomAccessFile = new RandomAccessFile(myFile, "rwd" ); 
myRandomAccessFile. seek(myPausedPosition); // 设 置 文件 写 入 点 


byte[ ] myBuffer = new byte[ 1024]; 

int myCount = 0, myLength; 

while ((myLength = myInputStream.read(myBuffer)) != -1){ 

if (!isPaused) { // 若 未 处 于 暂停 状态 , 则 正常 进行 下 载 操作 
myRandomAccessFile.write(myBuffer, 0, myLength); 
myCount += myLength; 


} else { // 若 处 于 暂停 状态 , 则 停止 文件 写 入 操作 
myPausedPosition = myCount; 
myRandomRccessFile.close( ); 

} 


Message myProgressMessage = new Messagel(); 
myProgressMessage.what = 1; 
myProgressMessage.argl = (int) ((((isPaused ? 0 : myPausedPosition) + 
myCount) / (float) fileSize) * 100); 
// 向 Handler 发 送 消息 并 更 新 进度 
myHandler. sendMessage( myProgressMessage); 
} 
myRandomAccessFile. close( ); 
if (myProgressBar. getProgress() == 100) myHandler. sendEmptyMessage(2); 
} } catch (Exception e) { 
Log.e("mytag", e.toString()); 
} } }). start(); 


上 面 这 上段 代码 在 MyCode\ MySample881 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 人 码 中 ,myRandomAccessFile. seek (myPausedPosition ) 用 于 根 
据 指 定 参 数 设 置 下 载 文件 的 写 入 位 置 。myRandomAccessFile. write(myBuffer, 0, myLength) 表 示 
以 0 为 起 始 位 置 , 从 myBuffer 写 人 myLength 的 数据 。RandomAccessFile 是 用 来 支持 随机 存 取 文件 
的 类 ,随机 存 取 文件 的 行为 就 像 存 取 在 文件 系统 里 一 个 很 大 的 字 节 数组 一 样 。 为 了 方便 存 取 文件 ,此 
字 节 数组 提供 了 “文件 指针 ”概念 : 类 似 于 游标 和 下 标 , 在 此 实例 中 , 则 使 用 该 指针 记录 和 暂停 的 下 载 位 
置 ,控制 下 载 文 件 的 读 写 , 从 而 实现 断 点 续 传 的 功能 。 此 外 ,访问 网 络 和 读 写 SD 卡 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 一"androlid。permission. 
INTERNET"/ > 和 < uses-permission android: name= "android. permission. WRITE_EXTERNAL._ 
STORAGE"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample881 文件 夹 中 ， 


226 ”使 用 HttpURLConnection 下 载 图 像 文 件 


此 实例 主要 通过 使 用 BitmapFactory 的 decodeStream() 方 法 解析 从 指定 网 络 地 址 获取 的 图 像 数 
据 流 ,实现 在 ImageView 控件 中 显示 指定 网 址 的 图 像 。 当 实例 运行 之 后 ,在 "图像 地 址 : ”输入 框 中 输 
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人 代表 一 个 图 像 的 网 络 地 址 ,如 “https://pl. ssl. qhmsg. com /t01992db65fb4747713. jpg”, 然 后 单 击 
“显示 图 像 ” 按 钮 , 则 在 下 面 显示 该 网 址 代表 的 图 像 , 效 果 分 别 如 图 226. 1 的 左 图 和 右 图 所 示 。 
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图 226. 1 


public class MainActivity extends Activity { 
EditText myEditText; 
ImageView myImageView; 


private Handler myHandler = new Handler() { // 在 主线 程 中 定义 一 个 Handler 对 像 
public void handleMessage(Message msg) { 
Bitmap myBmp = (Bitmap) msg.obj; // 把 Bitmap 显示 到 ImageView 控件 上 
myImageView. setImageBitmap(myBmp); 
} 
(Override 


protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myEditText = (EditText) findViewById(R. id. myEditText) ; 
myImageView = (ImageView) findViewById(R. id. myImageView) ; 
} 
// 响 应 单 击 " 显 示 图 像 "按钮 
public void onClickmyBtnl (View v) throws IOException { 
new Thread() { // 创 建 一 个 子 线程 
public void run() { 
try { 
String myPath = myEditText.getText().toString().trim(); 
URL myUrl = new URL(myPath); 
HttpURLConnection myHttpURLConnection = 
(HttpURLConnection) myUr1. openConnection( ); 
myHttpURLConnection. setConnectTimeout(5000); 
int myCode = myHttpURLConnection. getResponseCode( ); // 获 取 服 务 器 返回 状态 码 
if (myCode == 200) { 
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// 获 取 图 像 数据 ,不 管 是 什么 数据 都 是 以 流 的 形式 返回 

InputStream myStream = myHttpURLConnection. getInputStream( ) ; 

Bitmap myBmp = BitmapFactory. decodeStream(myStream) ; 

Message myMsg = Message.obtain(); 

myMsg. obj = myBmp; 

myHandler. sendMessage( myMsg); // 发 消息 把 Bitmap 显示 到 ImageView 控件 
} } catch (Exception e) { 

Toast. makeText (getApplicationContext( ), 

e.getMessage( ). toString(), Toast.LENGTH SHORT). show(); 
} } }. start(); 
}} 


上 面 这 段 代 码 在 MyCode\ MySample353\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myStream 二 myHttpURLConnection. getInputStream() 
用 于 根据 指定 的 网 址 获取 图 像 的 数据 流 ,myBmp 三 BitmapFactory. decodeStream(myStream) 用 于 
将 图 像 数 据 流 解析 为 位 图 格式 。 此 外 ,访问 指定 网 址 的 图 像 需 要 在 AndroidManifest. xml 文件 中 添 
加 < uses-permission android:name 二 "android. permission. INTERNET"/ > 权限。 此 实例 的 完整 项 目 
在 MyCode\MySample353 文件 夹 中 。 
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227 ”使 用 QuickContactBadge 访问 联系 人 


此 实例 主要 通过 使 用 QuickContactBadge 控件 的 assignContactFromPhone() 方 法 ,实现 快速 联 
络 联系 人 。 当 实例 运行 之 后 , 单 击 电话 图 标 , 如 图 227. 1 的 左 图 所 示 ,如果 电 话 号 码 指 定 的 联系 人 在 
通讯 录 中 已 经 存在 , 则 直接 显示 联系 人 信息 ; 否则 弹出 新 增 电话 号 码 窗口 ,如 图 227. 1 的 右 图 所 示 。 


MySample 


将 "13996060871" 添 加 到 通讯 录 ? 


图 227. 1 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
QuickContactBadge myQuickContactBadge = 
(QuickContactBadge) findViewById(R. id.myQuickContactBadge); 
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myQuickContactBadge. assignContactFromPhone("13996060871", 
false); // 为 联系 人 指定 一 个 电话 号 码 
| 


上 面 这 上段 代码 在 MyCode\ MySample067\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,assignContactFromPhone(String phoneNumber, boolean 
lazyLookup) 用 于 为 联系 人 指定 电话 号 码 , 其 中 ,参数 phoneNumber 表示 联系 人 的 电话 号 码 ; 参数 
lazyLookup 如 果 为 true, 将 不 会 立即 查找 这 个 电话 号 码 , 直 到 View 被 点 击 ; 否则 会 在 通讯 录 中 立即 
查找 这 个 电话 号 码 。 此 外 ,还 可 以 使 用 QuickContactBadge 的 assignContactFromEmail (String 
emailAddress， boolean lazyLookup) 方 法 指定 联系 人 的 电子 邮箱 地 址 ,该 方法 的 emailAddress 参数 表 
示 联 系 人 的 电子 邮箱 地 址 ,该 方法 的 lazyLookup 参数 如 果 为 true, 将 不 会 立即 查找 这 个 邮箱 地 址 , 直 
到 View 被 单 击 ; 否则 会 立即 查找 这 个 邮箱 地 址 。 此 实例 的 完整 项 目 在 MyCode\MySample067 文件 
夹 中 。 


228 ”使 用 ContentProviderOperation 增加 联系 人 


此 实例 主要 通过 使 用 ContentProviderOperation. newJInsert() 方 法 ,实现 在 当前 手机 的 通讯 录 中 
增加 联系 人 。 当 实例 运行 之 后 ,在 3 个 输入 框 中 分 别 输入 “联系 人 姓名 ”联系 人 电话 号 码 ” 和 “联系 人 
电子 邮箱 ”等 信息 ,然后 单 击 “在 当前 手机 的 通讯 录 中 增加 联系 人 ”按钮 ,如 果 增 加 成 功 , 则 在 弹出 的 
Toast 中 显示 增加 成 功 的 信息 ,如 图 228. 1 的 左 图 所 示 。 此 时 也 可 以 在 手机 的 通讯 录 中 查看 刚才 增加 
的 联系 人 信息 ,如 图 228. 1 的 右 图 所 示 。 


15058196637 


caixisa@163.com 


在 当前 手机 的 通讯 录 中 增加 联系 人 
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图 228.1 
主要 代码 如 下 : 


// 响 应 单 击 "在 当前 手机 的 通讯 录 中 增加 联系 人 "按钮 
public void onClickButtonl(View v) { 
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try { 
Contact myContact = new Contact(); 
myContact. setName(myEditName. getText().toString( )); 
myContact. setEmail(myEditEmail.getText().toString( )); 
myContact. setNumber (myEditPhone. getText().toString( )); 
addContact (myContact ); 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e. getMessage( ) . toString( ) Toast. LENGTH_ SHORT). show( ) ; 
1 
public String getContactID(String myName) { // 根 据 联系 人 姓名 查询 ID 
String myID = "0"; 
Cursor myCursor = getContentResolver().query( 
android. provider. ContactsContract. Contacts. CONTENT URI, 
new String[ ] {android. provider. ContactsContract. Contacts. _ID}, 
android. provider. ContactsContract. Contacts. DISPLAY NAME 
+ "='" + myName + ll nl 
if(myCursor. moveToNext()) { 
myID = myCursor.getString(myCursor.getColumnIndex( 
android. provider. ContactsContract. Contacts. ID) ); 
} 
return myID; 
} 
public void addContact(Contact myContact) { // 新 增 联系 人 
ArrayList < ContentProviderOperation > myData = 
new ArrayList < ContentProviderOperation >( ); 
String myID = getContactID(myContact. getName( ) ); 
if(!myID.equals("0")) { 
Toast. makeText (MainActivity. this, 
"联系 人 已 经 存在 ",， Toast.LENGTH SHORT). show( ) ; 
return; 
} else if(myContact. getName( ).trim().equals("")){ 
Toast. makeText (MainActivity. this, 
"联系 人 名 字 不 能 为 空 "，Toast.LENGTH SHORT). show( ); 
return; 
} else { 
myData. add( ContentProviderOperation. newInsert( 
ContactsContract. RawContacts. CONTENT URI) 
.WithValue( ContactsContract. RawContacts. ACCOUNT TYPE, null) 
.WithValue( ContactsContract. RawContacts. ACCOUNT NAME, null).build()); 
myData. add( ContentProviderOperation. newInsert( 
ContactsContract. Data. CONTENT URI) 
.withValueBackReference(COLUMN RAW CONTACT ID, 0) 
. withValue(COLUMN MIMETYPE, MIMETYPE STRING NAME) 
. withValue(COLUMN NAME, myContact.getName()).build()); // 增 加 姓名 
if(!myContact. getNumber().trim().equals("")) { 
myData. add( ContentProviderOperation. newInsert( 
ContactsContract. Data. CONTENT URI) 
.withValueBackReference(COLUMN RAW CONTACT ID, 0) 
. withValue(COLUMN MIMETYPE, MIMETYPE STRING PHONE) 
.WithValue(COLUMN NUMBER, myContact.getNumber()).build());  // 增 加 电话 
} 
if(!myContact. getEmail().trim().equals("")) { 
myData. add( ContentProviderOperation. newInsert( 
ContactsContract. Data. CONTENT URI) 
.withValueBackReference(COLUMN RAW CONTACT ID, 0) 
.withValue(COLUMN MIMETYPE, MIMETYPE STRING EMAIL) 
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.withValue(COLUMN EMAIL, myContact.getEmail()).build()); // 增 加 邮箱 
} 
try { 
ContentResolver myContentResolver = this. getContentResolver(); 
myContentResolver. applyBatch(ContactsContract. AUTHORITY, myData); 
Toast. makeText (MainActivity. this, 
"成 功 增加 联系 人 " + myContact. name, Toast.LENGTH SHORT). show( ) ; 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e.getMessage( ). toString(), Toast.LENGTH SHORT). show( ); 
}} 
} 


上 面 这 段 代 码 在 MyCode\MySample229\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 ,增加 联系 人 的 整个 操作 由 4 个 ContentProviderOperation. newlInsert 操作 组 
合 而 成 ,从 第 二 个 操作 开始 ,每 个 操作 都 有 一 个 withValueBackReference (COLUMN RAW _ 
CONTACT_ID, 0) 步 又 ,这 是 因为 它 参 照 了 第 一 个 操作 新 添加 联系 人 的 ID ,因为 是 批 处 理 , 插 和 人 数据 
前 并 不 知道 ID 值 ,在 进行 批 处 理 插入 数据 时 , 它 会 重新 引用 新 ID 值 , 因 此 不 会 影响 最 终 的 结果 。 此 
外 ,增加 手机 联系 人 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name = 
"android. permission. READ CONTACTS " /> 权限 和 < uses-permission android: name 三 "android. permission. 
WRITE_CONTACTS" /> 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample229 文件 夹 中 。 


229 ”使 用 ContentProviderOperation 修改 联系 人 


此 实例 主要 通过 使 用 ContentProviderOperation. newUpdate() 方 法 ,实现 在 当前 手机 的 通讯 录 
中 根据 联系 人 姓名 修改 电话 号 码 和 电子 邮箱 。 当 实例 运行 之 后 ,在 三 个 输入 框 中 分 别 输 入 “联系 人 姓 
名 “电话 号 码 ” 和 “电子 邮箱 ”等 信息 ,然后 单 击 “ 修 改 信 息 ” 按 钮 , 如 果 操 作成 功 , 则 在 弹出 的 Toast 中 
显示 操作 结果 ,如 图 229. 1 的 左 图 所 示 。 此 时 也 可 以 在 手机 的 通讯 录 中 查看 刚才 修改 的 联系 人 信息 ， 
如 图 229. 1 的 右 图 所 示 。 
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MySample 
联系 人 姓名 : 罗 斌 
电话 号 码 : ”13996060872 
电子 邮箱 : ”binluobin@163.com| 
修改 信息 


13996060872 
手机 


binluobin@163.com 
个 人 


修改 联系 人 信息 操作 成 功 ! 


图 229.1 
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public void onClickmyBtn1 (View v) { // 响 应 单 击 "修改 信息 "按钮 
Contact oldContact = new Contact( ); 
oldContact. setName( myEditName. getText( ) .toString() ); 
Contact newContact = new Contact(oldContact); 
newContact. setName( myEditName. getText().toString( )); 
newContact. setNumber(myEditPhone. getText().toString( )); 
newContact. setEmail(myEditEmail.getText().toString( )); 
updateContact(oldContact, newContact ); 
} 
public String getContactID(String myName) { // 根 据 联 系 人 姓名 查询 ID 
String myID = "0"; 
Cursor myCursor = getContentResolver().query( 
android. provider. ContactsContract. Contacts. CONTENT URI, 
new String[ ] {android. provider. ContactsContract. Contacts. ID)}, 
android. provider. ContactsContract. Contacts.DISPLAY NAME + 
"=" + myName + "'", null, null); 
if (myCursor.moveToNext()) { 
myID = myCursor.getString(myCursor.getColumnIndex( 
android. provider. ContactsContract. Contacts. ID) ); 
} 
return myID; 
} 
// 根 据 联 系 人 姓名 修改 其 电话 号 码 和 电子 邮箱 地 址 
public void updateContact(Contact oldContact, Contact newContact) { 
String myID = getContactID(oldContact. getName() ); 
if(myID.equals("0")) { 
Toast. makeText (MainActivity. this, 
oldContact. getName( ) + "不 存在 !" ，Toast.LENGTH SHORT). show( ); 
} else if(newContact. getName().trim().equals("")){ 
Toast. makeText (MainActivity.this, 
"联系 人 姓名 不 能 是 空白 ! "，Toast. LENGTH_SHORT). show( ); 
} else { 
ArrayList < ContentProviderOperation> myData = 
new ArrayList < ContentProviderOperation>(); 
// 修 改 联系 人 姓名 
myData. add( ContentProviderOperation. newUpdate( 
ContactsContract. Data. CONTENT URI).withSelection(COLUMN CONTACT ID 
+ "=? AND" + COLUMN MIMETYPE + " = ?" ,new String[ ]{myID, MIMETYPE STRING NAME)}) 
.withValue(COLUMN _ NAME，newContact. getName( ) ) .build() ); 
if(!newContact. getNumber( ).trim().equals("")) { 
// 修 改 电话 号 码 
myData. add( ContentProviderOperation. newUpdate( 
ContactsContract. Data. CONTENT URI).withSelection(COLUMN CONTACT ID+ 
"=? AND" + COLUMN MIMETYPE + "=?", 
new String[ ] {myID, MIMETYPE STRING PHONE)}) 
.withValue(COLUMN NUMBER, newContact.getNumber()).build()); 
} 
if(!newContact.getEmail().trim().equals("")) { 
// 修 改 电 子 邮 箱 地 址 
myData. add( ContentProviderOperation. newUpdate( 
ContactsContract. Data. CONTENT URI).withSelection(COLUMN CONTACT ID 
+"=? AND" + COLUMN MIMETYPE +" = ?" ,new String[ ]{myID, MIMETYPE STRING EMAIL}) 
.withValue(COLUMN EMAIL, newContact.getEmail()).build()); 
} 
try { 
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getContentResolver( ).applyBatch(ContactsContract. AUTHORITY, myData); 
Toast. makeText (MainActivity. this, 
"修改 联系 人 信息 操作 成 功 !"， Toast.LENGTH_SHORT). show( ); 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e.getMessage( ). toString(), Toast.LENGTH SHORT). show( ); 
} 
} 
} 


上 面 这 段 代 码 在 MyCode\MySample231\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 段 代 码 中 ,ContentProviderOperation. newUpdate (ContactsContract. Data. CONTENT 
URD. withSelection (COLUMN _CONTACT _ ID 十 "=? AND ”十 COLUMN _MIMETYPE 十 
"一 ?" ,new Stringl |{myID, MIMETYPE_STRING_ PHONE)). withValue (COLUMN_NUMBER ， 
newContact. getNumber()). build() 用 于 根据 联系 人 的 myID 修改 电话 号 码 , 联 系 人 的 myID 是 联系 
人 在 通讯 录 中 的 唯一 标识 ,此 处 则 是 通过 myID = getContactID (oldContact. getName()) 获 取 。 此 
外 ,修改 手机 联系 人 信息 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 
一 "android. permission. READ_CONTACTS" /> 权限 和 < uses-permission androlid:name 一 "android. 
permission. WRITE_CONTACTS"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample231 文件 夹 中 。 


230 ”使 用 ContentProviderOperation 删除 联系 人 


此 实例 主要 通过 使 用 ContentProviderOperation. newDelete() 方 法 ,实现 在 当前 手机 的 通讯 录 中 
根据 姓名 删除 此 联系 人 。 当 实例 运行 之 后 ,在 “联系 人 姓名 : ”输入 框 中 输入 姓名 ,如 “ 罗 斌 ”, 然 后 单 击 
“删除 联系 人 ”按钮 ,如果 操 作成 功 , 则 在 弹出 的 Toast 中 显示 操作 结果 ,否则 显示 联系 人 不 存在 ,效果 
分 别 如 图 230. 1 的 左 图 和 右 图 所 示 。 此 时 也 可 以 在 手机 的 通讯 录 中 查看 刚才 的 操作 结果 ，。 
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public void onClickmyBtn1 (View v) { // 响 应 单 击 " 删 除 联系 人 "按钮 
Contact myContact = new Contact(); 
myContact. setName( myEditName. getText().toString( )); 
deleteContact (myContact); 
} 
public String getContactID(String myName) { // 根 据 联系 人 姓名 查询 ID 
String myID = "0"; 
Cursor myCursor = getContentResolver().query( 
android. provider. ContactsContract. Contacts. CONTENT URI, 
new String[ ] {android. provider. ContactsContract. Contacts. ID)}, 
android. provider. ContactsContract. Contacts. DISPLAY NAME 
+"='" + myName + "'", null, null); 
if (myCursor.moveToNext()) { 
myID = myCursor.getString(myCursor.getColumnIndex( 
android. provider. ContactsContract. Contacts. ID)); 


} 
return myYID; 
| 
public void deleteContact(Contact myContact) { // 根 据 姓 名 删除 该 联系 人 
ArrayList < ContentProviderOperation > myData = 
new ArrayList < ContentProviderOperation >(); 
String myID = getContactID(myContact. getNanme( )); 
if(myID=="0"){ 
Toast. makeText (MainActivity.this, 
"联系 人 不 存在 !"，Toast. LENGTH_SHORT). show( ); 
return; 
} 
myData. add( ContentProviderOperation. newDelete( 
ContactsContract. RawContacts. CONTENT URI) 
.WithSelection(ContactsContract. RawContacts. CONTACT ID 
+"="+myID, null).build()); 
try { 
getContentResolver( ).applyBatch(ContactsContract. AUTHORITY, myData); 
Toast. makeText (MainActivity.this, 
"删除 联系 人 操作 成 功 !"， Toast.LENGTH_SHORT). show( ); 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e.getMessage().toString(), Toast.LENGTH SHORT). show( ); 
} } 


上 面 这 段 代码 在 MyCode\MySample232\app\src\main\java\com\bin\luo\mysample\ MainActivity. 
java 文件 中 。 在 这 上段 代码 中 , ContentProviderOperation. newDelete (ContactsContract. RawContacts. 
CONTENT_URD. withSelection (ContactsContract. RawContacts. CONTACT _IDT"=" + myID, null). 
build() 用 于 根据 联系 人 的 myID 删除 联系 人 ,联系 人 的 myID 是 联系 人 在 通讯 录 中 的 唯一 标识 ,此 处 
则 是 通过 myID = getContactID (myContact. getName()) 获 取 。 此 外 ,删除 手机 联系 人 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 王 "androlid. permission. READ 
CONTACTS"/> 权限 和 < uses-permission android: name 一 " android. permission. WRITE _ 
CONTACTS"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample232 文件 夹 中 。 
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231 ”使 用 ContentResolver 检测 飞行 模式 的 状态 


此 实例 主要 通过 在 Settings. System. getInt() 方 法 的 参数 中 使 用 ContentResolver 参数 和 
Settings. System. AIRPLANE_MODE_ON 参数 ,实现 检测 当前 手机 的 飞行 模式 设置 情况 。 当 实例 运 
行 之 后 , 单 击 "检测 飞行 模式 的 使 用 状态 ?按钮 ,如 果 在 手机 设置 中 局 用 飞行 模式 ,如 图 231. 1 的 左 图 
所 示 , 则 在 弹出 的 Toast 中 提示 “飞行 模式 已 局 用 1”, 如 图 231. 1 的 右 图 所 示 ,否则 在 弹出 的 Toast 中 
提示 “飞行 模式 已 禁用 !”。 
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飞行 模式 检测 飞行 模式 的 使 用 状态 


默认 短信 应 用 
信息 


网 络 共享 与 便携 式 热点 


VPN 


飞行 模式 已 启用 ! 


图 231. 1 


public void onClickBtnl(View v) { // 响 应 单 击 "检测 飞行 模式 的 使 用 状态 "按钮 
ContentResolver myContentResolver = getContentResolver( ); 
// 通 过 ContentResolver 获取 飞行 模式 启用 状态 标志 
int myFlag = Settings. System. getInt(myContentResolver, 
Settings. System. AIRPLANE MODE ON, 0); 
// 通 过 标志 判断 飞行 模式 是 否 启 用 
if (myFlag == 1) Toast.makeText(MainActivity.this, 
"飞行 模式 已 启用 !"，Toast. LENGTH_SHORT). show( ); 
else if (myFlag == 0) 
Toast. makeText (MainActivity. this, 
"飞行 模式 已 禁用 ! " ，Toast. LENGTH SHORT). show(); 


上 面 这 段 代 码 在 MyCode\ MySample896\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample896 文件 夹 中 。 
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232 ”使 用 ContentResolver 检测 手机 的 时 间 格 式 


此 实例 主要 通过 在 android. provider. Settings. System。 getString ( ) 方法 的 参数 中 使 用 
ContentResolver 参数 和 android. provider. Settings. System. TIME 12 _ 24 参数 ,实现 检测 当前 手机 的 
时 间 格 式 的 设置 情况 。 当 实例 运行 后 , 单 击 “ 检 测 当 前 手机 的 时 间 格 式 ” 按 钮 ,如 果 在 手机 设置 中 设置 
的 时 间 格 式 是 24 小 时 格式 ,如 图 232. 1 的 左 图 所 示 , 则 在 弹出 的 Toast 中 提示 “当前 手机 的 时 间 格 式 
是 : 24”, 如 图 232. 1 的 右 图 所 示 ,否则 在 弹出 的 Toast 中 提示 “当前 手机 的 时 间 格 式 是 : 12”。 
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图 232. 1 
主要 代码 如 下 : 


public void onClickBtnl(View v) { // 响 应 单 击 "检测 当前 手机 的 时 间 格 式 " 按 锂 
ContentResolver myContentResolver = getContentResolver( ) ; 
// 通 过 ContentResolver 获取 当前 手机 的 时 间 格 式 
String myFormat = android. provider. Settings. System. getString( 
myContentResolver, android. provider. Settings. System. TIME 12 24); 
Toast. makeText (MainActivity. this, 
"当前 手机 的 时 间 格 式 是 :" + myFormat, Toast. LENGTH SHORT) . show( ) ; 


上 面 这 段 代 人 码 在 MyCode\ MySample902\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample902 文件 夹 中 ，。 


233 ”使 用 ContentResolver 攻取 所 有 短信 


此 实例 主要 通过 使 用 getContentResolver() 的 query() 方 法 ,实现 查询 当前 手机 的 所 有 短信 内 容 。 
当 实 例 运行 之 后 , 单 击 “ 获 取 当 前 手机 的 所 有 短信 和 内容” 按钮 , 则 在 下 面 的 列表 中 显示 当前 手机 中 的 所 
有 短信 和 内容, 效果 分 别 如 图 233. 1 的 左 图 和 右 图 所 示 。 
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MySample MySample 


获取 当前 手机 的 所 有 短信 内 容 获取 当前 手机 的 所 有 短信 内 容 


OS dd 


手机 号 码 : 10086 短信 和 内容: 【 拒 把 活 】 请 
好 友 扫 码 助力 ， 最 高 得 3.4G 流 量 大 礼包 ! 好 友 
新 关注 并 绑 定 重庆 移动 公众 号 ， 也 可 领 1GB! 
赶快 进入 重庆 移动 公众 号 -我 的 优惠 -最 高 领 
3.4GB 生 成 您 的 专属 二 维 码 海报 吧 ! (如 不 愿 
ee 【中 国 移 


手机 号 码 : 10086 短信 和 内容: 尊敬 的 客户 ， 
截止 到 6 月 17 日 19 时 32 分 ， 您 的 话费 余额 : 
79.05 元 ， 女 点击 http://wap.cq.10086.cn/z/ 
zdcx 查看 每 月 账单 详情 。 

【 送 15 元 话费 】 现 在 为 您 赠送 3 个 月 咪 咕 游戏 
玩家 会 员 ， 回 复 51101 即 可 领取 (立即 生 

效 ) ，5 元 /月 ， 领 取 后 从 次 月 起 连续 3 个 月 赠 
送 5 元 话费 (业务 前 3 个 月 不 能 取消 ) ， 可 享受 
咪 咕 游戏 内 所 有 单机 道具 八 折 优惠 ， 每 周 可 接 
受 最 新 游戏 推荐 彩信 周刊 。 【中 国 移动 ] 


手机 号 码 : 10086 短信 内 容 : 102 


手机 号 码 : 10086 短信 内 容 : 尊敬 的 客户 ， 
您 本 月 的 套餐 余 量 如 下 : 

1. 通 话 分 钟 数 剩余 0 分 钟 ， 

2. 通 用 流量 剩余 0.0M， 专 属 流量 剩余 


eg 
图 233.1 
主要 代码 如 下 : 
public void onClickmyBtnl(View v) { // 响 应 单 击 " 获 取 当 前 手机 的 所 有 短信 内 容 " 按 钮 
try { 


ArrayList myArray = new ArrayList <>(); 
ContentResolver myContentResolver = getContentResolver( ); 
Cursor myCursor = myContentResolver.query(Uri.parse("content://sms/"), 
new String[ ]{" address", "body"}, null, null, "date desc" ); 
if (myCursor != null) { 
while (myCursor.moveToNext()) { 
String myPhone = myCursor.getString(myCursor.getColumnIndex("address" )); 
String myBody = myCursor.getString(myCursor.getColumnIndex(" body" )); 
String myInfo = ”手机 号 码 :”+ myPhone + "短信 内 容 :”+ myBody; 
myArray.add(myInfo); 
} 
myCursor. close( ); 
} 
myListView. setAdapter(new ArrayAdapter(this, 
android.R. layout. simple list item 1, myArray)); 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e.getMessage( ) .toString( ) ，Toast.LENGTH SHORT). show( ); 
} 


上 和 面 这 段 代码 在 MyCode\ MySample228\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,myContentResolver(). query (Uri. parse ("content:// 
sms/"), new String| |{"address", "body"}, null, null, "date desc" ) 用 于 查询 当前 手机 的 短信 信息 ， 
参数 Uri. parse("content://sms/") 表 示 从 内 容 提 供 者 中 查询 手机 的 短信 ; 参数 new String| ]{" 
address"，"body"} 表 示 在 查询 结果 中 返回 短信 的 内 容 和 电话 号 码 ; 第 三 个 参数 名 为 selection, 表 示 设 
置 查 询 条 件 , 相 当 于 SQL 语句 中 的 where,null 表示 不 进行 第 选 ; 第 四 个 参数 名 为 selectionArgs, 这 
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个 参数 需要 配合 第 三 个 参数 使 用 ,如 果 在 第 三 个 参数 里 面 有 问号 ,那么 在 selectionArgs 写 和 人 的 数据 就 
会 替换 邱 "?”, 如 果 第 三 个 参数 为 null, 则 此 参数 也 应 该 为 null; 第 五 个 参数 名 为 sortOrder, 表 示 按 照 
什么 要 求 进 行 排序 。 此 外 ,查询 手机 短信 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission 
android: name 一 "android. permission. READ _SMS"/ 人 > 权限。 此 实例 的 完整 项 目 在 MyCode\ 
MySample228 文件 夹 中 。 


234 ”使 用 ContentResolver 儿 取 通话 记录 


此 实例 主要 通过 在 getContentResolver() 的 query() 方 法 中 使 用 CallLog. Calls. CONTENT_URI 
参数 ,实现 查询 当前 手机 所 有 的 通话 记录 。 当 实例 运行 之 后 ,将 在 ListView 控件 中 显示 当前 手机 所 
有 的 通话 记录 ,效果 分 别 如 图 234. 1 的 左 图 和 右 图 所 示 ，。 


MySample MySample 


获取 当前 手机 的 通话 记录 获取 当前 手机 的 通话 记录 
未 备注 联系 人 


02388357958 2018-05-16 18-08-14 0 分 钟 打 入 


未 备注 联系 人 


15581469790 2018-05-15 19-52-55 0 分 钟 打 入 


未 备注 联系 人 


02389162125 2018-05-15 10-48-22 0 分 钟 打 入 


未 备注 联系 人 


12582 2018-05-11 20-26-40 0 分 钟 打 入 


图 234. 1 
主要 代码 如 下 : 


public void onClickButtonl(View v) { // 响 应 单 击 " 获 取 当 前 手机 的 通话 记录 "按钮 
List< Map < String，String > myData = getDataList( ) ; 
SimpleAdapter mySimpleAdapter = new SimpleAdapter(this, myData, 
R. layout. myitem, new String[ ]{"name", "number", "date", 
"duration", "type"}, new int[ ]{R. id.myName, R. id. myNumber, 
R. id. myDate, R. id.myDuration, R. id.myType}); 
myListView. setAdapter(mySimpleAdapter); 
} 
private List< Map< String, String>> getDataList() { 
try { 
ContentResolver myContentResolver = getContentResolver( ); 
myCursor = myContentResolver. query( 
CallLog. Calls. CONTENT_URI, // 通 话 记录 的 URI 
new String[ ] {CallLog. Calls. CACHED NAME, // 通 话 记录 的 联系 人 
CallLog. Calls. NUMBER, // 通 话 记 录 的 电话 号 码 
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CallLog. Calls. DATE, // 通 话 记 录 的 日 期 
CallLog. Calls. DURATION, // 通 话 时 长 
CallLog. Calls. TYPE)}, // 通 话 类 型 

null, null, CallLog. Calls. DEFAULT SORT ORDER); // 按 照 时 间 逆 序 排列 


} catch (SecurityException e) { } 
List <Map<= String, String >> myArrayList = new ArrayList < Map< String, String >>(); 
while (myCursor.moveToNext()) { 
String myName = 
myCursor. getString(myCursor. getColumnIndex(CallLog.Calls. CACHED NAME)); 
String myNumber = 
myCursor. getString(myCursor. getColumnIndex(CallLog. Calls. NUMBER ) ); 
long myDate = myCursor.getLong(myCursor.getColumnIndex(CallLog.Calls.DATE)); 
String newDate = 
new SimpleDateFormat("yyyy -~ MM- dd HH -~ mm - ss" ).format(new Date(myDate)); 
int myDuration = 
myCursor. getInt(myCursor.getColumnIndex(CallLog. Calls. DURATION) ); 
int myType = myCursor.getInt(myCursor.getColumnIndex(CallLog.Calls. TYPE)); 
String myCatagory = ""; 
switch (myType) { 
case CallLog. Calls. INCOMING TYPE: 
myCatagory = " 打 入 "; 
break; 
case CallLog. Calls. OUTGOING TYPE: 
myCatagory = "打出 "; 
break; 
case CallLog. Calls. MISSED TYPE: 
myCatagory = "未 接 "; 
break; 


Map < String, String> map = new HashMap < String, String >(); 
map. put(" name" ，(myName == null) ? "未 备注 联系 人 ": myName); 
map. put(" number", myNumber); 

map. put("date", newDate); 

map. put ("duration", (myDuration / 60) + "分 钟 " ); 

map. put(" type”", myCatagory); 

myArrayList. add( map); 


} 
return myArrayList; 
} 


上 面 这 段 代 码 在 MyCode\ MySample671 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySimpleAdapter 二 new SimpleAdapter(this, myData， 
R. layout. myitem, new String| |{"name", "number", "date", "duration", "type"}, new int| |{R. 
id. myName, R. id. myNumber, R. id. myDate, R. id. myDuration，R. id. myType}) ) 用 于 在 自 定 义 适 
配 帮 中 根据 myitem 布局 的 样式 设置 列表 项 的 对 应 字段 。 关 于 myitem 布局 的 详细 内 容 请 参考 源 代 码 
中 的 MyCode\MySample671\app\src\main\res\layout\myitem. xml 文件 。 此 外 ,查询 通话 记录 需要 
在 AndroidManifest. xml 文件 中 添加 < uses-permission androlid:name 王 "android. permission. READ_ 


CALL_LOG"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample671 文件 夹 中 。 
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235 ”使 用 ContentResolver 获取 SD 卡 的 文件 


此 实例 主要 通过 在 getContentResolver() 的 query() 方 法 中 指定 查询 参数 为 某 种 文件 类 型 ,从 而 
获取 手机 外 部 存储 介质 上 的 指定 类 型 的 所 有 文件 。 当 实例 运行 之 后 , 单 击 “ 获 取 SD 卡 上 的 所 有 MP3 
文件 ”按钮 , 则 在 ListView 控件 中 显示 手机 外 部 存储 上 的 所 有 MP3 文件 , 单 击 在 ListView 控件 中 的 
任意 列表 项 , 则 在 弹出 的 Toast 中 显示 该 文件 的 路 径 信息 ,如 图 235. 1 的 左 图 所 示 。 单 击 “ 获 取 SD 卡 
上 的 所 有 PNG 文件 "按钮 , 则 在 ListView 控件 中 显示 手机 外 部 存储 上 的 所 有 PNG 文件 , 单 击 在 
ListView 控件 中 的 任意 列表 选项 , 则 在 弹出 的 Toast 中 显示 该 文件 的 路 径 信 息 , 如 图 235. 1 的 右 图 


所 示 。 


MySample 


获取 SD 卡 上 的 所 有 MP 3 文件 
获取 SD 卡 上 的 所 有 PNG 文 件 


Opera 2 (歌剧 2) 7823991 
Born Too Slow 6675021 
Get Low 4287829 
41 268427 
43 290505 
42 267237 


e8a3493663f75a8d9bf6b400d 1393997 
0d919f5 


background0 461362 


主要 代码 如 下 : 


// 响 应 单 击 "获取 SD 卡 上 的 所 有 MP3 文件 "按钮 


MySample 
获取 SD 卡 上 的 所 有 MP3 文 件 
获取 SD 卡 上 的 所 有 PNG 文 件 


boy_body 826374 
boy_face 1210486 
pepsi_elements_boy 331612 
girlLd_2 924916 
girl_face_d 1419475 
pepsi_elements_gir| 732877 
.Shengdanlaoren8000001 172 
.Shengdanlaoren8000002 172 
/storage/sdcard0/qqmusic/arScene/ 
model/22/boy_face.png 


.shengdanlaoren8000005 172 


void OnClickBtn]l (View view){ GetFiles("audio/mpeg"); } 


// 响 应 单 击 "获取 SD 卡 上 的 所 有 PNG 文件 "按钮 


void OnClickBtn2(View view){ GetFiles("image/png"); } 


private void GetFiles(String myType) { 
myFiles. clear( ); 
mySizes. clear( ); 
myItems. clear( ); 
myPathData. clear( ); 


// 通 过 ContentResolver 查询 指定 外 部 存储 文件 信息 
ContentResolver myContentResolver = getContentResolver(); 


Cursor myCursor = myContentResolver. query( 


MediaStore. Files.getContentUri ("external” ), null, 
"mime type = \"" +myType+"\"", null, null); 


while (myCursor.moveToNext()) { 


String myName = myCursor.getString(myCursor.getColumnIndex( 


MediaStore. Files. FileColumns. TITLE) ); 


// 显 示 名 称 
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String mySize = myCursor. getString(myCursor. getColumnIndex( 
MediaStore. Files. FileColumns. SIZE) ); // 显 示 文 件 大 小 
byte[ ] data = myCursor.getBlob(myCursor.getColumnIndex( 
MediaStore. Files. FileColumns. DATA) ) ; // 文 件 的 位 置 数据 
myFiles.add(myName); 
mySizes.add(mySize); 
// 将 位 置 数据 转换 成 字符 串 形式 的 路 径 
String myPath = new String(data, 0, data. length - 1); 
myPathData. add( myPath); 


} 


for (int i = 0; i< myFiles. size(); i++) { 
HashMap < String, Object > myListItem = new HashMap <>(); 
myListItem.put("name", myFiles.get(i)); 
myListItem. put ("size", mySizes.get(i)); 
myItems.add(myListItem); 


} 


SimpleAdapter myAdapter = new SimpleAdapter(MainActivity.this, myItems, 
R. layout. myitem, new String[ ]{" name”, "size” }, 
new int[ ]{R. id. myName, R. id. mySize} ); 

myListView. setAdapter(myAdapter); 

// 单 击 文件 名 称 ( 列 表 项 ) 显 示 文 件 路 径 信息 

myListView. setOnItemClickListener(new RdapterView. OnItemClickListener() { 


(Override 


public void onItemClick(AdapterView <?> parent,， 


View view, int position, long id){ 


Toast. makeText (MainActivity. this, 


} }); 
} 


上 面 这 段 代 码 在 MyCode\ MySample672\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 上 面 这 段 代 码 中 ,myAdapter 一 new SimpleAdapter (MainActivity. 
this，myItems，R. layout. myitem, new String[ |{"name", "size"} ,new int| |{R. id. myName, R. id. 
mySize) ) 表 示 根 据 myitem 布局 的 样式 创建 ListView 的 列表 项 ,关于 myitem 布局 的 详细 内 容 请 参考 
源 代码 中 的 MyCode\MySample672\app\src\main\res\ layout\myitem. xml 文件 。myCursor = 
myContentResolver. query( MediaStore. Files. getContentUri("external”"), null,"mime type 一 \”" 十 


myType 十 \"", null, null) 表 示 在 手机 外 部 存储 介质 上 查找 mime_type 为 指定 文件 类 型 的 文件 。 文 


myPathData. get (position), Toast. LENGTH SHORT). show( ); 


\ 


件 类 型 和 文件 后 级 名 有 相应 的 映射 关系 ,常用 的 文件 后 缀 名 的 映射 关系 如 下 。 


(1) MimeTypeMap. 
(2) MimeTypeMap. 
(3) MimeTypeMap. 
(4) MimeTypeMap. 
(5) MimeTypeMap. 
(6) MimeTypeMap. 
(7) MimeTypeMap. 
(8) MimeTypeMap. 
(9) MimeTypeMap. 


loadEntry("application/pd{" , "pdf"), 
loadEntry("application/zip" , "zip" )。 
loadEntry("application/msword", "doc"), 
loadEntry("application/vnd. ms-excel", "xls" ), 
loadEntry("application/vnd. ms-powerpoint" , "ppt" )。 
loadEntry("audio/midi", "mid" ), 
loadEntry("audio/mpeg", "mp3")。 
loadEntry("audio/x-pn-realaudio”", "rm" ), 


loadEntry("audio/x-wWav", "wav" )。 


(10) MimeTypeMap. loadEntry("image/bmp", "bmp"), 
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(11) MimeTypeMap. loadEntry("image/gif", "gif")., 
(12) MimeTypeMap. loadEntry("image/ico" , "ico" ), 
(13) MimeTypeMap. loadEntry("image/ijpeg" , "jpg" )。 
(14) MimeTypeMap. loadEntry("image/png", "png")。 


(15) MimeTypeMap. loadEntry("image/svg+ xml", "svg" )。 
(16) MimeTypeMap. loadEntry("image/tif{f" , “tif")., 

(17) MimeTypeMap. loadEntry("image/x-photoshop" , psd ) 。 
(18) MimeTypeMap. loadEntry("text/plain", "text"”) 。 


(19) MimeTypeMap. loadEntry("video/mpeg", "mpeg" )。 

(20) MimeTypeMap. loadEntry("video/mp4", "mp4"), 

(21) MimeTypeMap. loadEntry("video/x-ms-wmv", "wmv" ), 

(22) MimeTypeMap. loadEntry("video/x-msvideo" , ”avi ) 。 

此 外 , 读 取 存储 卡 上 的 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: 
name 一 "android. permission. READ EXTERNAL STORAGE"/ > 权限 。 需 要 说 明 的 是 ,使 用 此 实例 
的 相关 类 需要 在 gradle 中 引入 compile 'com. android. support: design: 25. 0.1' 依 赖 项 。 此 实例 的 完 


整 项 目 在 MyCode\MySample672 文件 夹 中 ， 


236 ”使 用 ContentResolver 改变 屏 芝 沈 度 值 


此 实例 主要 实现 了 使 用 手指 在 屏幕 上 进行 上 下 滑动 , 增 大 或 减 小 屏幕 的 亮度 。 当 实例 运行 后 ,如 
朱 手 指 回 下 滑动 , 则 亮度 减 小 ; 如 果 手 指 癌 上 滑动 , 则 亮度 增 大 ,效果 分 别 如 图 236. 1 的 左 图 和 右 图 所 
示 。 注 意 : 此 实例 在 部 分 模拟 右 中 测试 无 效 , 因 此 应 在 手机 上 直接 测试 ; 虽然 实例 屏幕 截图 看 不 出 明 
显 的 亮度 变化 ,但 在 手机 上 直接 观察 时 变化 非常 明显 。 
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public void setBrightness( int myBrightness) { 


} 


WindowManager. LayoutParams myParams = getWindow().getAttributes( ); 
myParams. screenBrightness = Float.valueOf (myBrightness) * (1f / 255f); 
getWindow( ) . setAttributes(myParams); 
Uri myUri = android. provider. Settings.System.getUriFor(" screen brightness" ); 
ContentResolver myResolver = getContentResolver( ); 
android. provider. Settings. System. putInt (myResolver, 
"screen brightness", myBrightness); // 改 变 亮 度 值 
getContentResolver().notifyChange(myUri, null); 


public int getScreenBrightness() { 


} 


int currentBrightness = 0; 
ContentResolver myResolver = getContentResolver( ); 
try { 
currentBrightness = android.provider.Settings.System.getInt( 
myResolver, Settings.System.SCREEN BRIGHTNESS); // 获 取 亮 度 值 
} catch (Exception e) { } 
return currentBrightness; 


上 面 这 上段 代码 在 MyCode\ MySample718\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,操作 手机 设置 需要 在 AndroidManifest. xml 文件 中 添加 
< uses-permission android:name 一 "android. permission. WRITE SETTINGS" /> 权限 。 此 实例 的 完 


整 项 目 在 MyCode\MySample718 文件 夹 中 ， 


237 ”使 用 ContentResolver 设置 屏 有 化 沈 度 值 


此 实例 主要 通过 ContentResolver 修改 数据 库 的 screen_brightness 字段 ,实现 改变 手机 的 屏幕 亮 
度 。 当 实例 运行 之 后 ,如 果 回 右 拖 动 滑 块 , 则 屏幕 亮度 增 大 ; 如 果 回 左 拖 动 滑 块 , 则 屏幕 亮度 减 小 , 效 
果 分 别 如 图 237. 1 的 左 图 和 右 图 所 示 。 注 意 : 此 实例 应 在 Android 5.0 以 上 的 手机 中 测试 ,在 模拟 天 


中 测试 可 能 无 效 。 


MySample MySample 


改变 屏幕 亮度 :一 @ 


图 237. 1 
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主要 代码 如 下 : 


// 根 据 指 定 值 设置 亮度 
public void setBrightness(Activity activity, int brightness) { 
WindowManager. LayoutParams myParams = 


activity. getWindow( ) .getAttributes( ); // 获 取 当 前 窗口 属性 参数 对 象 
myParams. screenBrightness = 
Float. valueOf (brightness) * (1f/255f); // 设 置 亮度 值 


activity. getWindow( ) . setAttributes(myParams); 

// 如 果 不 更 新 系统 , 则 退出 应 用 后 ,亮度 改变 立即 失效 

Uri myUri = android. provider. Settings. System. getUriFor(" screen brightness" ); 
ContentResolver myContentResolver = getContentResolver( ); 


// 根 据 新 亮度 值 修改 数据 库 对 应 字段 
android. provider. Settings. System. putInt (myContentResolver, 

"screen brightness" ,brightness); 
myContentResolver. notifyChange( myUri, null); // 通 知 更 新 手机 


上 和 面 这 段 代 码 在 MyCode\ MySample817\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 外 ,修改 系统 设置 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission android:name 二 "android. permission. WRITE_SETTINGS"/> 权 限 。 此 实例 的 完整 项 日 
在 MyCode\MySample817 文件 夹 中 ， 


238 ”使 用 ContentResolver 检测 诞 转 屏 臣 功能 


此 实例 主要 通过 在 Settings. System. getInt() 方 法 的 参数 中 使 用 ContentResolver 参数 和 
android. provider. Settings. System. ACCELEROMETER_ROTATION 参数 ,实现 检测 当前 手机 的 自 
动 旋转 屏幕 功能 的 设置 情况 。 当 实例 运行 之 后 , 单 击 "检测 是 否 启 用 目 动 旋转 屏幕 功能 ”按钮 ,如 果 在 
手机 设置 中 已 经 启用 自动 旋转 屏幕 功能 按钮 (注意 : 在 实现 横 屏 和 竖 屏 自动 切换 时 ,特别 需要 检测 此 
设置 ) ,如 图 238. 1 的 左 图 所 示 , 则 在 弹出 的 Toast 中 提示 “已 启用 自动 旋转 屏幕 1”, 如 图 238. 1 的 右 
图 所 示 ,否则 在 弹出 的 Toast 中 提示 “已 禁用 自动 旋转 屏幕 !1”。 


en &% = 


MySample 


检测 是 否 启用 自动 旋转 屏幕 功能 


已 启用 自动 旋转 屏幕 ! 


已 作为 媒体 设备 连接 
二 办 中 不 其 人 也 USB 碗 . 


dS Np 下 


图 238. 1 


>》 Android 炫 酷 应 用 300 例 . 实战 篇 


主要 代码 如 下 : 


// 响 应 单 击 "检测 是 否 启用 自动 旋转 屏幕 功能 "按钮 
public void onClickBtnl (View v) { 


ContentResolver myContentResolver = getContentResolver( ) ; 
// 通 过 ContentResolver 获取 自动 旋转 屏幕 功能 标志 
int myFlag = Settings. SYstem. getInt(myContentResolver， 
android. provider. Settings. System. ACCELEROMETER ROTATION, 0); 


if (myFlag == 1) // 通 过 标志 判断 自动 旋转 屏幕 功能 是 否 启用 
Toast. makeText (MainActivity. this, 


"已 启用 自动 旋转 屏幕 !"，Toast. LENGTH_SHORT). show( ); 
else if (myFlag == 0) 


Toast. makeText (MainActivity. this, 


"已 禁用 自动 旋转 屏幕 !"，Toast. LENGTH_SHORT). show(); 
} 


上 面 这 段 代码 在 MyCode\ MySample900\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample900 文件 夹 中 。 


239 ”使 用 BroadcastReceiver 监听 来 电 电 话 号 码 


此 实例 主要 通过 使 用 广播 接收 者 BroadcastReceiver, 实 现 监 听 手 机 来 电 电 话 号 码 。 当 实例 运行 


之 后 ,从 其 他 手机 向 当前 手机 拨打 电话 , 则 在 弹出 的 Toast 中 显示 来 电 方 的 电话 号 码 , 如 图 239. 1 
所 示 。 


上 年 11;03 


六 来 电 己 码 :13996060872 


239. 1 
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主要 代码 如 下 : 


public class CallReceiver extends BroadcastReceiver { 
(Override 
public void onRecelive(Context context, Intent myIntent) { 
String myPhone = 
myIntent. getExtras().getString(TelephonyManager. EXTRA INCOMING NUMBER); 
Toast. makeText (context, "新 来 电 号 码 :" + myPhone, Toast. LENGTH_SHORT). show( ) ; 
上 


上 F 面 这 段 代 码 在 MyCode\ MySample285 \app\src\main\java\com\bin\luo\mysample\ 
CallReceiver. java 文件 中 。 在 这 段 代 码 中 ,myPhone 二 myl]ntent. getExtras ( ). getString 
(TelephonyManager. EXTRA_INCOMING_NUMBER) 用 于 获取 手机 来 电 电 话 号 码 , 但 它 主要 是 通 
过 广播 接收 者 BroadcastReceiver 的 onReceive () 方 法 以 参数 的 形式 传递 的 ,因此 通常 需要 以 
BroadcastReceiver 为 基 类 创建 一 个 新 的 广播 接收 者 ,并 且 需 要 在 AndroidManifest. xml 文件 中 进行 
注册 和 添加 权限 ,注册 的 主要 代码 如 下 : 


< receiver android:name = " .CallReceiver”>> 
< intent - 人 filter > 
< action android: name = "android. intent. action. PHONE_STRTE”/> 
< action android: name = "android. intent. action. NEW OUTGOING CALL" /> 
</intent - filter > 
</receiver > 


添加 权限 的 主要 代码 如 下 : 


< uses - permission android:name = "android. permission. READ PHONE STATE" /> 
< uses - permission android:name = "android. permission. PROCESS OUTGOING CALLS"/> 


上 面 这 段 代 码 在 MyCode\MySample285\app\src\main\AndroidManifest. xml 文件 中 。 此 实例 
的 完整 项 目 在 MyCode\MySample285 文件 夹 中 。 


240 使 用 BroadcastReceiver 判断 手机 电池 是 人 否 正 在 充电 


此 实例 主要 通过 使 用 registerReceiver() 方 法 注册 电池 服务 请 求 , 实 现 以 广播 接收 的 形式 判断 手 
机 是 否 在 充电 。 当 实例 运行 之 后 ,如 果 手 机 正在 充电 , 则 将 每 隔 一 定时 间 在 弹出 的 Toast 中 显示 “ 手 
机 正在 充电 ...”, 如 图 240. 1 的 左 图 所 示 。 如 打手 机 没有 充电 , 则 将 每 隔 一 定时 间 在 弹出 的 Toast 中 
显示 “手机 未 充电 1!”, 如 图 240. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 

Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout. activity main); 
IntentFilter myFilter = new IntentFilter(Intent.ACTION BATTERY CHANGED); 
registerReceiver(MyPowerConnectionReceiver, myFilter); 

} 


<》 anaroia 红 本 应 300 . 实战 篇 


private BroadcastReceiver MyPowerConnectionReceiver = new BroadcastReceiver() { 
(Override 
public void onRecelive(Context context, Intent intent) { 
int myStatus = intent.getIntExtra(BatteryManager. EXTRA STATUS, - 1); 
boolean isCharging = myStatus == BatteryManager.BATTERY STATUS CHARGING || 
myStatus == BatteryManager. BATTERY STATUS FULL; 
if (isCharging) { 
Toast. makeText (getApplicationContext( ), 
"手机 正在 充电 ...",， Toast. LENGTH_SHORT). show( ); 
} else { 
Toast. makeText (getApplicationContext( ), 
"手机 未 充电 !" ，Toast.LENGTH_SHORT) . show( ); 


> 


MySample MySample 


于 机 正在 充电 . 手机 未 充电 ! 


图 240. 1 


上 F 面 这 段 代 码 在 MyCode\ MySample292\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,int myStatus 三 intent. getIntExtra (BatteryManager. 
EXTRA_STATUS, 一 1) 用 于 接收 系统 以 广播 形式 发 送 的 手机 电池 状态 信息 。boolean isCharging 
= myStatus 三 一 BatteryManager. BATTERY STATUS _ CHARGING| |myStatus 三 三 BatteryManager. 
BAITERY_STAIUS FULL 用 于 判断 当前 手机 电池 是 否 正在 充电 。 如 果 需 要 持续 不 断 地 接收 系统 以 
广播 形式 发 送 的 手机 电池 状态 信息 , 则 应 该 在 AndroidManifest. xml 文件 中 添加 ACTION_POWER_ 
CONNECTED 和 ACTION_POWER_DISCONNECTED ,主要 代码 如 下 : 


< receiver android:name = " .MyPowerConnectionReceiver"> 
< intent - filter> 
< action android:name = "android. intent.action. ACTION POWER CONNECTED" /> 
< action android:name = "android. intent. action. ACTION POWER DISCONNECTED" /> 
</intent - filter > 


</receiver > 
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上 面 这 段 代 码 在 MyCode\MySample292\app\src\main\AndroidManifest. xml 文件 中 。 此 实例 
的 完整 项 目 在 MyCode\MySample292 文件 夹 中 ， 


241 使 用 BroadcastReceiver 监听 屏 幕 开 启 或 关闭 


此 实例 主要 通过 使 用 广播 接收 者 监听 Intent. ACTION_SCREEN_ON 和 Intent. ACTION _ 
SCREEN_OFF ,实现 监视 手机 屏幕 打开 或 关闭 。 当 实例 运行 后 ,在 “音乐 文件 : ”输入 框 中 输入 在 SD 
卡 上 存储 的 音乐 文件 名 称 ,然后 单 击 “开始 播放 ?按钮 ,将 播放 指定 的 音乐 文件 ,如 图 241. 1 所 示 。 此 
时 如 果 关 闭 屏 幕 ,将 停止 播放 音乐 ; 如 果 打 开 屏 幕 , 则 将 继续 播放 音乐 。 


MySample 


乐 文 件 :;dcard0/mymusic.mp3 开始 播放 


习 241.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

EditText myFile; 

MediaPlayer myPlayer; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout.activity main); 
myFile = (EditText) findViewById(R. id. myFile); 
myFile. setText(Environment. getExternalStorageDirectory!() 

+ "/mymusic.mp3"); // 存 储 在 SD 卡 上 的 音乐 文件 mymusic. mp3 

myPlayer = new MediaPlayer(); 
IntentFilter myFilter = new IntentFilter(); 
myFilter.addAction( Intent. ACTION SCREEN OFF); 
myFilter.addAction( Intent. ACTION_ SCREEN_ON); 
registerReceiver(myReceiver, myFilter); 


依 Android 炫 酷 应 用 300 例 。 实战 篇 


public void onClickmyBtnl (View v) { // 响 应 单 击 " 开 始 播放 "按钮 
try { 
myPlayer. setDataSource(myFile. getText().toString( ) ) ; 
myPlayer. Prepare( ) ; 
myPlayer. start( ); 
} catch (Exception e) { e.printStackTrace( ); } 
} 
private final BroadcastReceiver myReceiver = new BroadcastReceiver() { 
(Override 
public void onReceive(final Context myContext，final Intent myIntent) { 
final String myRction = myIntent.getRction( ); 
if (Intent. ACTION SCREEN ON. equals(myRction)) { 


myPlayer. start( ); // 如 果 打 开 屏 幕 , 就 继续 播放 
} else if (Intent.ACTION SCREEN OFF.equals(myAction)) { 

myPlayer. pause( ); // 如 果 关 闭 屏 幕 , 就 停止 播放 
1 


上 面 这 段 代 码 在 MyCode\ MySample278\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myFilter. addAction(Intent. ACTION_ SCREEN _OFF) 
和 myFilter. addAction(Intent. ACTION_SCREEN_ON) 用 于 在 IntentFilter 中 添加 打开 屏幕 和 关闭 
屏幕 这 两 个 动作 ,然后 通过 registerReceiver(myReceiver，myFilter) 回 系统 提交 接收 这 两 个 动作 的 请 
求 。 当 发 生 关闭 或 打开 屏幕 的 动作 时 ,系统 就 把 通知 发 给 myReceiver，Intent. ACTION_SCREEN_ 
ON. equals(myAction) 则 根据 这 两 个 动作 决定 当前 应 用 应 该 做 什么 。 此 外 , 读 取 在 SD 卡 上 的 音乐 文 
件 需 要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android:name 二 "android. permission. 
READ_EXTERNAL_STORAGE"/ > 权限。 此 实例 的 完整 项 目 在 MyCode\MySample278 文件 夹 中 。 


242 自 定 义 BroadcastReceiver 实现 短信 拦截 


此 实例 主要 通过 自 定义 BroadcastReceiver ,实现 短信 拦截 功能 。 当 实例 运行 后 , 单 击 “ 开 局 短信 
拦截 功能 ”按钮 , 则 当前 应 用 处 于 短信 拦截 监听 状态 ; 单 击 * 向 10086 发 送 查 询 短信 "按钮 , 则 跳 转 到 短 
信 应 用 界面 ,然后 单 击 右 侧 的 发 送 按钮 ( 右 箭头 ), 如 图 242. 1 的 左 图 所 示 ; 当 10086 收 到 查询 请 求 之 
后 , 则 立即 回 当 前 手机 发 送 短信 ,此 时 当前 应 用 就 会 将 10086 发 送 的 短信 拦截 下 来 ,如 图 242. 1 的 右 
图 所 示 。 


主要 代码 如 下 : 

public void onClickBtnl1(View v) { // 响 应 单 击 "开启 短信 拦截 功能 "按钮 
IntentFilter myFilter = new IntentFilter(); // 动 态 设 置 过 滤 信 息 
myFilter.addRction("android. provider. Telephony. SMS_RECEIVED" ); 
registerReceiver(new SmsReceiver(), myFilter); // 动 态 注册 广播 接收 者 


Toast. makeText (MainActivity. this, 
"短信 拦截 功能 已 开启 !"，Toast. LENGTH_SHORT). show( ); 

} 
public void onClickBtn2(View v) { // 响 应 单 击 "向 10086 发 送 查 询 短 信 " 按 钮 

Intent myIntent = new Intent(Intent.ACTION SENDTO, Uri.parse("smsto:10086")); 

myIntent. putExtra("sms_body", "101"); 

startActivity(myIntent); // 跳 转 到 短信 应 用 界面 进行 短信 发 送 操作 
} 
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public class SmsReceiver extends BroadcastReceiver { 
(Override 
public void onReceive(Context context, Intent intent){ 
if(intent.getAction() == "android. provider. Telephony. SMS RECEIVED" ){ 
Object[ ] myPdus = (Object[ ] ) intent. getExtras().get(" pdus" ); 
ArrayList myArrayList = new ArrayList(); 
for (Object myPdu: myPdus){ 
SmsMessage myMessage = SmsMessage. createFromPdul( (byte[ ] )myPdu); 
String myAddress = myMessage. get0riginatingAddress();  // 获 取 发 送 人 号 码 
String myBody = myMessage. getMessageBody!( ); // 获 取 短 信和 内 容 
String myDate = new SimpleDateFormat ("yyyy - MM- dd HH: mm: ss" ). 
format(new Date(myMessage. getTimestampMillis( ))); // 获 取 短 信和 发 送 时 间 
myRrrayList.add(" 短信 发 件 人 :" + myhddress+ " ,短信 内 容 :" 


+ myBody + " ,短信 发 送 时 间 :" + myDate); // 拼 接 短信 字符 串 , 并 封装 至 列表 
} 
// 输 出 所 拦截 的 短信 内 容 
MainRctivity. myTextView. setText(myArrayList. toString( ) ); 
}}} 


Q 回 时 a omnes 
€ 10086 NE Mysample 


en 开启 短信 拦截 功能 。 向 10086 发 送 查 询 短信 
[短信 发 件 人 : 10086 ,短信 内 容 : 尊敬 
101 的 客户 : 您 未 办 理 任何 套餐 信息 。 【中 
i 国 移动 】 ,短信 发 送 时 间 : 2018-04-07 
18:20:01] 


尊敬 的 客户 : 您 未 办 理 任 何 套餐 
信息 。 【中 国 移动 】 


下 午 6:19 


尊敬 的 客户 : 您 未 办 理 任何 套餐 
信息 。 【中 国 移动 】 


下 午 620 


@ 101 


图 242.1 


上 和 面 这 上段 代码 在 MyCode\ MySample880\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 外 ,操作 短信 和 需要 在 AndroidManifest. xml 文件 中 添加 权限 < uses- 
permission android: name 一 "android. permission. RECEIVE _ SMS"/>。 此 实例 的 完整 项 目 在 
MyCode\MySample880 文件 夹 中 。 


243 ”使 用 RingtoneManager 设置 手机 章 钟 铃声 


此 实例 主要 通过 使 用 RingtoneManager 的 setActualDefaultRingtoneUri() 方 法 ,实现 设置 手机 的 
闹钟 铃声 。 当 实例 运行 后 , 单 击 * 设 置 阅 钟 铃声 ?按钮 ,将 从 下 加 上 ee 
口 ,如 图 243. 1 的 左 图 所 示 ; 单 击 “ 仅 此 一 次 ”按钮 , 则 将 显示 “音乐 ”窗口 ,在 其 中 任意 选择 一 个 音乐 文 


<》 Anaroia 红 本 应 300 。 实战 篇 


件 , 该 文件 即 成 为 手机 的 曾 钟 铃声 ,如 图 243. 1 的 右 图 所 示 。 当 在 手机 “设置 "窗口 中 调整 “ 曾 钟 音量 ” 
时 , 即 可 听 到 刚才 设置 的 曾 钟 铃声 。 


a NE: 


alarm02.wav 

2 KD i -ay 

alarm01.wav 

4719.9KB 201 ey 
OO mymusic.mp3 


使 用 文件 管理 完成 操作 


使 用 其 他 应 用 
油 。 媒体 存储 
图 243.1 
主要 代码 如 下 : 
public void onClickmyBtnl(View v) { // 响 应 单 击 "设置 闹钟 铃声 "按钮 


Intent myIntent = new Intent(RingtoneManager. ACTION RINGTONE PICKER); 
myIntent. putExtra(RingtoneManager. EXTRA RINGTONE TYPE, 

RingtoneManager. TYPE ALARM); 
myIntent. putExtra(RingtoneManager. EXTRA RINGTONE TITLE," 设 置 闹钟 铃声 " ); 
startActivityForResult(myIntent, 1); 


} 
(Override // 设 置 铃 声 之 后 的 回调 函数 


protected void onActivityResult(int requestCode, int resultCode, Intent data) { 
super. onActivityResult(requestCode, resultCode, data); 
try { 
Uri myUri = data. getParcelableExtra(RingtoneManager. EXTRA RINGTONE PICKED URI); 
if (myUri != null) { 
switch (requestCode) { 
case 1: 
RingtoneManager. setActualDefaultRingtoneUri(this, 
RingtoneManager. TYPE ALARM, myUri); 
Toast. makeText (MainActivity. this, 
"铃声 设置 成 功 , 请 设置 闹钟 测试 !"，Toast. LENGTH_SHORT). show( ) ; 
break; 
default: 
break.; 
} } } catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e. getMessage( ) .toString( ) Toast. LENGTH SHORT). show( ); 


} } 
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上 面 这 上段 代码 在 MyCode\ MySample423\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myUri = data. getParcelableExtra (RingtoneManager. 
EXTRA_RINGTONE _PICKED _ URI) 用 于 在 选择 音乐 文件 之 后 返回 该 音乐 文件 的 Uri。 
Ringtone Manager. setActualDefaultRingtoneUri(this, Ringtone Manager. TYPE ALARM,., myUri) 
用 于 将 myUri 代表 的 音乐 文件 设置 为 出 钟 铃声 。 此 外 ,设置 铃声 需要 在 AndroidManifest. xml 文件 
中 添加 < uses-permission android:name 一 "androlid. permission. WRITE SETTINGS"/ > 权限 。 此 实 
例 的 完整 项 目 在 MyCode\MySample423 文件 夹 中 。 


244 使 用 RingtoneManager 设置 手机 通知 铃声 


此 实例 主要 通过 使 用 RingtoneManager 的 setActualDefaultRingtoneUri() 方 法 ,实现 设置 手机 的 
通知 提示 铃声 。 当 实例 运行 后 , 单 击 “设置 通知 铃声 ”按钮 ,将 从 下 同上 滑 出 “使 用 文件 管理 完成 操作 ? 
窗口 ,如 图 244. 1 的 左 图 所 示 ; 单 击 “ 仅 此 一 次 ”按钮 , 则 将 显示 “音乐 ”窗口 ,在 其 中 任意 选择 一 个 音乐 
文件 ,该 文件 即 成 为 手机 的 通知 提示 铃声 ,如 图 244. 1 的 右 图 所 示 。 当 在 手机 “设置 ”窗口 中 调整 “ 铃 
声音 量 ” 时 , 即 可 听 到 刚才 设置 的 通知 铃声 。 


“™ 站 52% 下 午 2:45 


alarm02.wav 


323.3KB 2017-10-29 


alarm01.wav 
479.9KB 2017-10-29 
OO mymusic.mp3 


使 用 文件 管理 完成 操作 


使 用 其 他 应 用 
而 。 媒体 存储 
图 244. 1 
主要 代码 如 下 : 
public void onClickmyBtnl(View v) { // 响 应 单 击 "设置 通知 铃声 "按钮 


Intent myIntent = new Intent(RingtoneManager. ACTION RINGTONE PICKER); 
myIntent. putExtra(RingtoneManager. EXTRA RINGTONE _ TYPE， 
RingtoneManager. TYPE NOTIFICATION); 
myIntent. putExtra(RingtoneManager. EXTRA RINGTONE TITLE, "设置 通知 铃声 " ); 
startActivityForResult (myIntent, 2); 
} 


@Override // 设 置 铃 声 之 后 的 回调 函数 
protected void onActivityResult(int requestCode, int resultCode, Intent data) { 


super. onActivityResult(requestCode, resultCode, data); 


16》 Andoid 权 上 ao 例 。 实战 篇 


try { 
Uri myUri = data. getParcelableExtra(RingtoneManager. EXTRA RINGTONE PICKED URI); 
if (myUri != null) { 
switch (requestCode) { 
case 2: 
RingtoneManager. setActualDefaultRingtoneUri(this, 
RingtoneManager. TYPE NOTIFICATION, myUri); 
Toast. makeText (MainActivity. this, 
"铃声 设置 成 功 , 请 设置 铃声 音量 测试 !" ，Toast. LENGTH_SHORT). show( ); 


break; 
} } } catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e. getMessage( ) .toString(), Toast.LENGTH_ SHORT). show( ) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySample424\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myUri 一 data. getParcelableExtra (RingtoneManager. 
EXTRA_RINGTONE _ PICKED _ URI) 用 于 在 选择 音乐 文件 之 后 返回 该 音乐 文件 的 Uri。 
Ringtone Manager. setActualDefaultRingtoneUri(this, Ringtone Manager. TYPE_NOTIFICATION， 
myUri) 用 于 将 myUri 代表 的 音乐 文件 设置 为 通知 提示 铃声 。 此 外 ,设置 伶 声 需要 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android: name 一 "android. permission. WRITE 
SETTINGS"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample424 文件 夹 中 。 


245 使 用 AlarmManager 以 指定 时 间 执 行 操作 


此 实例 主要 实现 了 使 用 AlarmManager 配合 Intent, 定 时 通过 Broadcast 发 送 提示 。 当 实例 运行 
之 后 ,在 弹出 的 Toast 中 提示 “10 秒 后 alarm 开启 ”, 如 图 245. 1 的 左 图 所 示 ; 然后 退出 当前 实例 ,等 待 
10 秒 , 将 收 到 广播 服务 ,并 在 弹出 的 Toast 提示 “ 啊 应 alarm”, 如 图 245. 1 的 右 图 所 示 。 


本 中 二 


MySample 


10 秒 后 alarm 开 启 3 响应 alarm 的 册 
mm 2 


图 245.1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
Intent myIntent = new Intent(MainActivity.this, 


AlarmReceiver. class); // 发 送 一 个 广播 ,广播 接收 后 Toast 提示 定时 操作 
myIntent. setAction("myFlag" ); 
PendingIntent myPendingIntent = 


PendingIntent. getBroadcast(MainActivity.this, 0, myIntent, 0); 
AlarmManager myAlarmManager = (AlarmManager)getSystemService(ALARM SERVICE); 
myAlarmManager. set(AlarmManager. RTC WAKEUP, 
System. currentTimeMillis() + 10 * 1000, myPendingIntent ); 
Toast. makeText (MainActivity. this, 
"10 秒 后 alarm 开启 " ，Toast. LENGTH_LONG). show( ); 
上 


上 面 这 段 代 码 在 MyCode\ MySample662\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myAlarmManager. set(AlarmManager.RTC_WAKEUP， 
System. currentTimeMillis( ) 十 10 x 1000，myPendingIntent) 表 示 10 秒 后 唤醒 AlarmManager。 
myIntent 一 new Intent(MainActivity. this，AlarmReceiver. class) 表 示 使 用 AlarmReceiver 广播 接收 
者 啊 应 AlarmManager 的 唤醒 服务 。AlarmReceiver 广播 接收 者 的 主要 代码 如 下 : 


public class AlarmReceiver extends BroadcastReceiver { 
(Override 
public void onReceive(Context context, Intent intent) { 


if(intent.getAction().equals("myFlag" )){ 
Toast. makeText (context, "响应 alarm" ，Toast.LENGTH LONG). show( ); 


} }} 


上 面 这 段 代 码 在 MyCode\ MySample662\app\src\main\java\com\bin\luo\mysample\ 
AlarmReceiver. java 文件 中 。 此 外 ,使 用 AlarmReceiver 广播 接收 者 需要 在 AndroidManifest. xml 文 
件 中 注册 ,主要 代码 如 下 : 


< receiver android:name =" .AlarmReceiver'" /> 


此 实例 的 完整 项 目 在 MyCode\MySample662 文件 夹 中 。 


246 使 用 AudioManager 获取 和 设置 音量 


此 实例 主要 通过 在 SeekBar 控件 的 onProgressChanged 事件 响应 方法 中 调用 AudioManager 的 
setStreamVolume() 方 法 ,实现 以 拖 动 SeekBar 滑 块 的 方式 改变 手机 音量 的 大 小 。 在 实例 运行 之 前 ， 
先 使 用 任 一 音乐 播放 器 播放 歌曲 ; 当 实例 运行 之 后 ,SeekBar 控件 的 滑 块 位 置 即 决定 当前 音量 的 大 
小 ,向 右 拖 动 滑 块 ,音量 变 大 ; 向 左 拖 动 滑 块 ,音量 变 小 ,如 图 246. 1 的 左 图 和 右 图 所 示 。 模 拟 禹 和 手 
机 实测 效果 均 相 同 。 

主要 代码 如 下 : 
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J 
er 1 


MySample MySample 


图 246. 1 


public class MainActivity extends Activity { 
public AudioManager myAudioManager; 
private int maxVolume, currentVolume; 
(@ Override 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreatel( savedInstanceState); 
setContentView(R. layout. activity main); 
myAudioManager = (AudioManager)getSystemService(Context.AUDIO SERVICE); 
// 获 取 系 统 最 大 音量 
maxVolume = myAudioManager.getStreamMaxVolume(AudioManager. STREAM MUSIC ) ; 
SeekBar mySeekBar = (SeekBar) findViewById(R. id.mySeekBar); 
mySeekBar. setMax( maxVolunme); // 设 置 拖 动 条 最 高 值 与 系统 最 大 音量 匹配 
// 获 取 音 量 当 前 值 
currentVolume = myAudioManager.getStreamVolume( AudioManager. STREAM MUSIC); 
mySeekBar. setProgress(currentVolume); 
mySeekBar. setOnSeekBarChangeListener(new SeekBar. OnSeekBarChangeListener() { 
// 音 量 大 小 改变 时 调用 
public void onProgressChanged( SeekBar arg0, int progress, boolean fromUser) { 
// 根 据 当 前 滑 块 值 设置 音量 大 小 
myAudioManager. setStreamVolume( AudioManager. STREAM MUSIC, progress, 0); 
} 
(Override 
public void onStartTrackingTouch( SeekBar seekBar) { } 
(Override 
public void onStopTrackingTouch(SeekBar seekBar) { } 
});}} 


上 面 这 段 代 码 在 MyCode\ MySample063\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上段 代码 中 ,myAudioManager. setStreamVolume (AudioManager. 
STREAM_ MUSIC, progress，0) 用 于 根据 SeekBar 的 当前 值 改 变 音 量 的 大 小 。 此 外 ,在 使 用 
AudioManager 调节 音量 大 小 时 ,必须 获取 系统 的 最 大 音量 , 即 maxVolume 二 myAudioManager. 


第 8 章 ”系统 和 设备 


getStreamMaxVolume ( AudioManager. STREAM MUSIC)。 此 实例 的 完整 项 目 在 MyCode\ 
MySample063 文件 夹 中 。 


247 ”使 用 PowerManasger 实现 屏 比 一 直 党 着 


此 实例 主要 通过 创建 PowerManager. WakeLock 的 实例 ,实现 在 应 用 运行 时 保持 屏幕 一 直 亮 着 。 
当 实 例 运行 后 ,即使 在 手机 上 设置 了 “屏幕 延 时 ”时 间 , 屏 幕 也 会 一 直 亮 着 ,如 图 247. 1 所 示 。 


贸 出 中 © Oh 画 ll 章 75% 上 午 11:39 


MySample 


注意 : 如 果 此 应 用 不 关闭 ， 屏 幕 将 一 直 亮 着 ! 


图 247. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
PowerManager myPowerManager = null; 
PowerManager. WakeLock myWakeLock = null; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
this. myPowerManager = 
(PowerManager)this. getSystemService(Context. POWER SERVICE); 
this.myWakeLock = 
this.myPowerManager. newWakeLock( PowerManager. FULL WAKE LOCK, ""); 
} 
(Override 
protected void onResume( ) { 
super. onResume( ) ; 
this. myWakeLock. acquire( ); 
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} 
(Override 
protected void onPause() { 


super. onPause( ); 
this. myWakeLock. release( ); 


} } 


上 面 这 上段 代码 在 MyCode\ MySample273\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 人 代码 中 , this. myWakeLock = 三 this myPowerManager. 
newWakeLock(PowerManager. FULL WAKE LOCK.,"") 用 于 创建 PowerManager. WakeLock 实 
例 ,this. myWakeLock. acquire() 用 于 保持 一 直 亮 着 的 状态 。 此 外 ,使 用 PowerManager. WakeLock 
需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 一 "androlid. permission. 


WAKE LOCK"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample273 文件 夹 中 。 
248 ”使 用 WallpaperManasger 设置 壁纸 


此 实例 主要 通过 使 用 壁纸 管理 器 WallpaperManager, 实 现 将 选中 的 图 像 设 置 为 壁纸 。 当 实例 运 
行 之 后 , 单 击 图 像 即 可 显示 下 一 幅 图 像 , 当 显示 到 图 库 的 最 后 一 幅 图 像 时 ,自动 跳 转 显 示 第 一 幅 图 像 ， 
效果 如 图 248. 1 的 左 图 所 示 。 单 击 “ 设 置 手机 壁纸 ”按钮 , 则 当前 显示 的 图 像 自 动 设置 为 手机 壁纸 , 效 
果 如 图 248. 1 的 右 图 所 示 。 


—— 


— 成 功 设置 手机 壁纸 ! 


主要 代码 如 下 : 


public void onClickButtonl (View v) { // 响 应 单 击 "设置 手机 壁纸 "按钮 
WallpaperManager myWallpaperManager = 
WallpaperManager. getInstance(this); // 获 取 壁 纸 管理 器 
BitmapDrawable myBitmapDrawable = 
(BitmapDrawable) myImageView. getDrawable( ); // 获 取 图 像 资 源 
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try { // 根 据 图 像 资 源 设置 壁纸 


myWallpaperManager. setBitmap(myBitmapDrawable. getBitmap( ) ); 
Toast. makeText (getApplicationContext(), 
"成 功 设置 手机 壁纸 !"，Toast.LENGTH_SHORT) . show( ); 
} catch (Exception e) { e.printStackTrace(); } 
} 


上 F 面 这 段 代 码 在 MyCode\ MySample060\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myWallpaperManager. setBitmap (myBitmapDrawable. 
getBitmap()) 用 于 将 ImageView 控件 的 当前 图 像 设 置 为 壁纸 。 除 了 可 以 使 用 setBitmap() 方 法 设置 
壁纸 外 ,常用 的 设置 壁纸 方法 还 有 setResource()、setWallpaper() 等 。 此 外 ,设置 手机 壁纸 需要 在 


AndroidManifest. xml 文件 中 添加 < uses-permission android: name 三 "android. permission. SET_ 


WALLPAPER"/> 权 限 。 此 实例 的 完整 项 目 在 MyCode\MySample060 文件 夹 中 ，。 


249 ”使 用 PackageManasger 获取 支持 分 享 的 应 用 
此 实例 主要 通过 使 用 PackageManager 类 的 queryIntentActivities() 方 法 ,获取 当前 手机 安装 的 


支持 分 享 功 能 的 所 有 应 用 信息 。 当 实例 运行 之 后 , 单 击 “ 获 取 手 机 中 支持 分 享 功能 的 应 用 ”按钮 , 则 在 
弹出 的 Toast 中 显示 当前 手机 安装 的 支持 分 享 功能 的 应 用 包 名 信息 ,如 图 249. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


获取 手机 中 支持 分 享 功能 的 应 用 获取 手机 中 支持 分 享 功能 的 应 用 


当前 手机 中 安装 的 支持 分 享 功能 的 应 
用 如 下 : 
com.android.mms 
com.android.bluetooth 
com.tencent.mtt 
com.tencent.mobileqq 
com.jingdong,app.mal 
com.tencent.map 
com.tencent.qqpimsecure 
com.tencent.mobileqq 
com.tencent.mobileqq 
com.tencentmm 
com.tencent.mm 
com.tencent.androidqqmail 
com.tencent.androidqqgmail 
com,tencent.androidqqmail 
com.baidu.netdisk 
com.baidu.netdisk 
com.sina.weibo 


图 249.1 
主要 代码 如 下 : 


// 响 应 单 击 "获取 手机 中 支持 分 享 功能 的 应 用 "按钮 

public void onClickmyBtnl (View v) { 
String myInfo = "当前 手机 中 安装 的 支持 分 享 功能 的 应 用 如 下 :"; 
Intent myIntent = new Intent(Intent.ACTION SEND, null); 
myIntent.addCategory( Intent. CATEGORY DEFAULT); 
myIntent. setType(" text/plain" ); 
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PackageManager myPackageManager = this. getPackageManager( ); 
List < ResolveInfo > myResolveInfos = 


myPackageManager. queryIntentActivities(myIntent, 
PackageManager. COMPONENT ENABLED STATE DEFAULT); 
for (int i = 0; i< myResolveInfos. size(); i++) { 
ResolveInfo myResolveInfo = myResolveInfos. get(i); 
myInfo += "\n" + myResolveInfo. activityInfo. packageName; 


} 
Toast. makeText (getApplicationContext( ), 


myInfo, Toast.LENGTH SHORT). show( ); 


上 F 面 这 段 代 码 在 MyCode\ MySample209\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,Intent mylIntent 二 new Intent(Intent. ACTION_SEND， 
null) 用 于 创建 支持 分 享 功 能 的 Intent。List < ResolveInfo > myResolveInfos 一 myPackageManager. 
queryIntentActivities(mylIntent, PackageManager. COMPONENT ENABLED_STATE DEFAULT) 用 于 
根据 myIntent 查找 所 有 符合 分 享 功 能 的 应 用 包 。 此 实例 的 完整 项 目 在 MyCode\MySample209 文件 
夹 中 。 


250 ”使 用 WifiManager 开启 或 关闭 WiFi 信 号 


此 实例 主要 通过 使 用 WifiManager 的 setWifiEnabled() 方 法 ,实现 启用 WiFi 网 络 信 号 或 关闭 
WiFi 网 络 信号 。 当 实例 运行 之 后 , 单 击 “打开 WiFi” 按 钮 , 则 将 启用 WiFi 网 络 连 接 信和 号, 单 击 * 关 闭 


或 关闭 WiFi 信号 有 一 个 时 间 过 程 ,因此 需要 在 延迟 一 定 的 时 间 后 才能 检测 到 正确 的 执行 结果 。 


MySample 


打开 WiFi 关闭 WiFi 


当前 WiFi 状 态 : WiFi 正 在 启动 中 当前 WiFi 状 态 : WiFi 正 在 关闭 中 


图 250. 1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
WifiManager myWifiManager; 
Handler myHandler = new Handler(); 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 
myWifiManager = (WifiManager) this.getSystemService(Context.WIFI SERVICE); 
myHandler. postDelayed(new Runnable() { 
(@ Override 
public void run() { 
checkWifiState( ); 
} jr1500); 


// 响 应 单 击 " 打 开 WiFi" 按 钮 
public void onClickmyBtnl (View v) { 
myWifiManager. setWifiEnabled( true); 
// 打 开 WiFi 有 个 时 间 过 程 ,因此 需要 延迟 一 定 的 时 间 进 行 检测 
myHandler. postDelayed(new Runnable() { 
(Override 
public void run() { 
CheckWifiState( ); 
} e1500)s 
} 
// 啊 应 单 击 " 关 闭 WiFi" 按 钮 
public void onClickmyBtn2(View v) { 
myWifiManager. setWifiEnabled(false); 
// 关 闭 WiFi 有 个 时 间 过 程 , 因此 需要 延迟 一 定 的 时 间 进 行 检测 
myHandler. postDelayed(new Runnable() { 
(Override 
public void run() { 
checkWifiState( ); 
} },1500); 
} 
public void checkWifiState() { 
String myInfo = "当前 WiFi 状态 :"; 
Switch (myWifiManager. getWifiState()) { 
case WifiManager. WIFI STATE DISABLED: 
myInfo+= "WiFi 已 经 关闭 "; 
break; 
case WifiManager. WIFI STATE DISABLING: 
myInfo+= "WiFi 正在 关闭 中 "; 
break; 
case WifiManager. WIFI STATE ENABLED: 
myInfo+= "WiFi 已 经 启用 "; 
break; 
case WifiManager. WIFI STATE ENABLING: 
myInfo+= "WiFi 正在 启动 中 "; 
break; 
case WifiManager. WIFI STATE UNKNOWN: 
myInfo+= "未 知 WiFi 状态 "; 
break; 
} 
Toast. makeText (MainActivity.this,myInfo, Toast.LENGTH SHORT). show( ); 
} 
} 
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上 上面 这 段 代 码 在 MyCode\ MySample263\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,setWifiEnabled() 用 于 启用 或 禁用 WiFi, 当 此 方法 的 参数 
值 为 true 时 ,表示 启用 WiFi; 当 此 方法 的 参数 值 为 false 时 ,表示 禁用 WiFi。 此 外 ,操作 WiFi 需要 相 
应 的 权限 ,主要 代码 如 下 : 


< uses - permission android:name = "android. permission. ACCESS WIFI STATE"/> 
< uses - permission android:name = "android. permission. ACCESS NETWORK STATE"/> 
< uses - permission android:name = "android. permission.CHANGE NETWORK STATE" /> 
< uses - permission android:name = "android. permission.CHANGE WIFI STATE"/> 


上 和 面 这 段 代 码 在 MyCode\MySample263\app\src\main\AndroidManifest. xml 文件 中 。 此 实例 
的 完整 项 目 在 MyCode\MySample263 文件 夹 中 。 


251 使 用 WifiManager 锋 取 IP 地 址 


此 实例 主要 通过 使 用 WifiInfo 的 getIpAddress() 方 法 ,实现 以 WiFi 管理 服务 的 方式 来 获取 IP 
地 址 。 当 实例 运行 之 后 , 单 击 “通过 WiFi 管理 服务 获取 IP 地 址 ”按钮 , 则 在 弹出 的 Toast 中 显示 IP 
地 址 ,如 图 251. 1 的 左 图 和 右 图 所 示 。 注 意 : 由 于 启用 或 关闭 WiFi 信号 有 一 个 时 间 过 程 ,因此 需要 
在 延迟 一 定 的 时 间 后 才能 检测 到 正确 的 执行 结果 ， 


MySample MySample 


通过 WiFi 管 理 服务 获取 IP 地 址 通过 WiFi 管 理 服务 获取 IP 地 址 


当前 的 IP 是 : 192.168.1.148 


图 251. 1 


Private String ConvertIP( int myIntIP) { 
return ( (myIntIP >> 0) &OxFF ) + "." +( (myIntIP >> 8)&OxFF ) + "." 
+ ( (myIntIP >> 16) &OxFF ) + "." +( (myIntIP >> 24) &OxFF ) ; 
} 
// 响 应 单 击 "通过 WiFi 管理 服务 获取 IP 地 址 "按钮 
public void onClickmyBtnl (View v){ 
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WifiManager myWifiManager = 
(WifiManager) getSystemService(Context. WIFI SERVICE); // 获 取 WiFi 管理 服务 
if (!myWifiManager. isWifiEnabled()) { // 判 断 WiFi 是 否 开启 
// 开 启 WiFi, 需要 设置 权限 :android. permission. CHANGE WIFI STATE 
myWifiManager. setWifiEnabled(true); 
Toast. makeText (MainActivity. this, 
"正在 开启 WiFi, 请 稍 后 重新 获取 IP 地 址 "，Toast. LENGTH_SHORT). show( ); 


// 获 取 WiFi 信息 ,需要 设置 权限 :android. permission. ACCESS WIFI STATE 


WifiInfo myWifiInfo = myWifiManager.getConnectionInfo( ); 
int myIntIP = myWifiInfo.getIpAddress(); // 获 得 整 型 IP 地 址 
String myStringIP = ConvertIP(myIntIP); // 将 整 型 IP 地 址 转换 为 字符 串 


Toast. makeText (MainActivity.this, 
"当前 的 IP 是 :" + myStringIP, Toast.LENGTH SHORT). show(); 


上 和 面 这 上段 代码 在 MyCode\ MySample350\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myIntIP = myWifiInfo. getIpAddress() 用 于 获取 IP 地 
址 。 此 外 ,操作 WiFi 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android: name 二 
"android. permission. ACCESS WIFI_STATE"/> 权 限 和 < uses-permission android: name 三 " 
android. permission. CHANGE_WIFI_STATE "/ > 权限 。 此 实例 的 完整 项 目 在 MyCode\MySample350 文 
件 夹 中 ， 


252 ”使 用 ConnectivityManager 判断 网 络 状 态 


此 实例 主要 通过 使 用 ConnectivityManager 类 的 getNetworkInfo() 方 法 ,从 而 判断 当前 手机 的 网 
络 连 接 状 态 。 当 实例 运行 之 后 , 单 击 “判断 当前 手机 的 网 络 连 接 状 态 ? 按 钮 ,如 果 当 前 手机 网 络 使 用 的 
是 数据 连接 , 则 在 弹出 的 Toast 中 显示 "正在 使 用 移动 数据 连接 网 络 !”, 如 图 252. 1 的 左 图 所 示 ; 如 果 
当前 手机 网 络 使 用 的 是 WiFi 连接 , 则 在 弹出 的 Toast 中 显示 “正在 使 用 WiFi 连接 网 络 1”, 如 图 252.1 
的 右 图 所 示 。 

~ Td" dB 上 午 1103 | Q 中 dx 可 94% 上 午 11:03 
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判断 当前 手机 的 网 络 连 接 状 态 判断 当前 手机 的 网 络 连 接 状 态 


正在 使 用 移动 数据 连接 网 络 ! 正在 使 用 WIFI 连接 网 络 ! 


图 252. 1 


> Android 炫 酷 应 用 300 例 ， 实战 篇 


主要 代码 如 下 : 


// 响 应 单 击 " 判 断 当 前 手机 的 网 络 连接 状态 "按钮 
public void onClickmyBtnl (View v) { 
Context myContext = this.getApplicationContext(); 
ConnectivityManager myConnectivityManager = (ConnectivityManager) 
myContext. getSystemService(Context. CONNECTIVITY SERVICE); 
if (myConnectivityManager == null){ return; } 
else { 
NetworkInfo myMobileNet = myConnectivityManager. 
getNetworkInfo(ConnectivityManager. TYPE MOBILE); 
NetworkInfo myWifiNet = myConnectivityManager. 
getNetworkInfo(ConnectivityManager. TYPE WIFI); 
if (myWifiNet. isConnected( ) ){ 
Toast. makeText (MainActivity. this, 
"正在 使 用 WiFi 连接 网 络 !"，, Toast. LENGTH_SHORT). show( ); 
}else if(myMobileNet. isConnected( ) ){ 
Toast. makeText (MainActivity. this, 
"正在 使 用 移动 数据 连接 网 络 !"，,Toast. LENGTH_SHORT). show( ); 
}else if(!myMobileNet. isConnected() 到 !myWifiNet. isConnected( )){ 
Toast. makeText (MainActivity. this, 
"当前 没有 网 络 连接 !" ,Toast.LENGTH_SHORT). show( ); 
} 


上 面 这 段 代 码 在 MyCode\ MySample215\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ，myMobileNet 一 myConnectivityManager. 
getNetworkInfo(ConnectivityManager. TYPE_ MOBILE) 用 于 获取 移动 数据 连接 信息 ,该 方法 的 参数 
支持 的 类 型 包括 : TYPE_WIFI、TYPE_BLUETOOTH TYPE_DUMMY TYPE_ETHERNET、 
TYPE_MOBILE、TYPE_MOBILE_DUN、TYPE_VPN、TYPE_WIMAX, 因 此 可 以 创建 多 个 不 同类 
型 的 网 络 连 接 进 行 测试 。myMobileNet. isConnected() 用 于 判断 移动 数据 网 络 连 接 是 否 成 功 ,如 果 连 
接 成 功 , 则 返回 true, 和 否则 返回 false。 此 外 ,检测 网 络 连接 需要 在 AndroidManifest. xml 文件 中 添加 
< Uses-permission android: name = "android. permission. ACCESS_NETWORK STATE"/ > 权限。 
此 实例 的 完整 项 目 在 MyCode\MySample215 文件 夹 中 。 


253 ”使 用 BluetoothAdapter 打开 或 关闭 监 牙 


此 实例 主要 通过 使 用 getSystemService() 方 法 获取 系统 蓝牙 管理 吉 BluetoothManager 的 适 配 寓 
BluetoothAdapter ,然后 调用 BluetoothAdapter 的 enable() 和 disable() 方 法 ,实现 动态 启用 或 禁用 手 
机 的 蓝牙 功能 。 当 实例 运行 之 后 , 单 击 “打开 蓝牙 ”按钮 , 则 将 启动 蓝牙 ,并 跳 转 到 系统 的 蓝牙 配对 设 
置 界面 ,如 图 253. 1 的 左 图 所 示 。 单 击 “ 关 闭 蓝 牙 ” 按 钮 , 则 将 关闭 蓝牙 ,如 图 253. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
BluetoothAdapter myBluetoothAdapter; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout.activity main); 
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// 通 过 系统 服务 获取 蓝牙 管理 器 
BluetoothManager myBluetoothManager = (BluetoothManager) 
getSystemService(Context. BLUETOOTH SERVICE); 

// 通 过 蓝牙 管理 器 获取 蓝牙 适配器 
myBluetoothAdapter = myBluetoothManager. getAdapter( ); 

} 
// 啊 应 单 击 " 打 开 蓝 牙 "按钮 

public void onClickBtnl (View v) { 
myBluetoothAdapter. enable( ); // 启 用 蓝牙 功能 
Intent myIntent = new Intent(Settings. ACTION BLUETOOTH SETTINGS); 
startActivity(myIntent); // 跳 转 至 蓝牙 设置 页 面 , 以 配对 蓝牙 
Toast. makeText(this, "蓝牙 已 打开 !",Toast.LENGTH_SHORT). show( ); 

} 
// 响 应 单 击 "关闭 蓝牙 "按钮 

public void onClickBtn2(View v) { 
myBluetoothAdapter. disable( ); // 禁 用 蓝牙 功能 
Toast. makeText(this," 蓝牙 已 关闭 !",Toast.LENGTH SHORT) . show( ); 

国 


MySample 


未 在 附近 找到 蓝牙 设备 。 


在 HLA Note3 上 开启 蓝牙 设置 后 ， 附 近 的 设备 将 可 以 


mm = 
人- 


蓝牙 已 关闭 ! 


图 253. 1 


系统 和 设备 


上 F 面 这 段 代 码 在 MyCode\ MySample874\app\src\ main\java\com\bin\luo\mysample\ 


254 ”使 用 LocationListener 获取 当前 经 纬度 值 


MainActivity. java 文件 中 。 需 要 注意 的 是 ,局 用 或 禁用 蓝牙 需要 在 AndroidManifest. xml 文件 中 添 
加 < uses-permission android: name 一 "androlid。permission。BLUETOOTH "> 权限 和 < 志 uses- 
permission android:name 一 "android. permission. BLUETOOTH _ ADMIN "/ > 权限 。 此 实例 的 完整 项 
日 在 MyCode\MySample874 文件 夹 中 。 


此 实例 主要 通过 在 LocationListener 类 的 onLocationChanged() 方 法 中 监听 位 置信 息 , 获 取 当 前 


手机 的 经 度 和 纬度 值 。 当 实例 运行 之 后 ,将 显示 当前 手机 的 经 度 值 和 纬度 值 ,如 图 254. 1 所 示 。 注 


Android 炫 酷 应 用 300 例 。 实战 篇 


意 : 在 测试 时 ,一 定 要 先 开 局 GPS, 并 且 尽 可 能 在 无 遮挡 的 空旷 地 方 ,否则 可 能 不 能 显示 数据 。 
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当前 手机 的 位 置 如 下 : 
经 度 :106.59395333333332 
纬度 :29.69044666666667 


图 254. 1 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

private LocationManager myLocationManager; 

private LocationListener myLocationlistener; 

String myLocationprovider; 

private TextView myTextView; 

(@Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity _ main) ; 


myTextView = (TextView) this.findViewById(R. id. myTextView) ; 


try { 
Criteria myCriteria = new Criterial( ) ; 
myCriteria. setAccuracy(Criteria. ACCURACY FINE); // 设 置 精度 
myCriteria. setAltitudeRequired(false); // 不 提供 海拔 高 度 信息 
myCriteria. setBearingRequired(false); // 不 提供 方向 信息 
myCriteria. setCostAllowed(true); // 人 允许 运营 商 计 费 
myCriteria. setPowerRequirement (Criteria. POWER LOW); // 设 置 电 池 消 耗 为 低 耗 费 
myLocationManager = 
(LocationManager) getSystemService(Context.LOCATION SERVICE); 
if (checkgps()) { // 检 查 GPS 功能 开启 
myLocationprovider = myLocationManager.getBestProvider(myCriteria, true); 
myLocationlistener = new MYLocationListener(); // 注 册 位 置 监 听 器 


if (Build. VERSION. SDK INT > = Build.VERSION CODES.M) { 
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if (checkSelfPermission(Manifest. permission. ACCESS FINE LOCATION) 
!= PackageManager. PERMISSION GRANTED & checkSelfPermission( 
Manifest. permission. ACCESS COARSE LOCATION) 
!= PackageManager.PERMISSION GRANTED) { return; } 
} 
myLocationManager. requestLocationUpdates( 
myLocationprovider, 1000, 0, myLocationlistener); 
} 
}catch(Exception e){ 
Toast. makeText (MainActivity. this, 
"异常 错误 :" + e.toString( ) ,Toast. LENGTH LONG) . show( ) ; 
} } 
Private class MyLocationListener implements LocationListener{ 
// 若 位 置 发 生变 化 , onLocationChanged 方法 被 调用 
(QOverride 
public void onLocationChanged( Location myLocation) { 
if(myLocation != null){ 
String myLatitude = Double.toString(myLocation.getLatitude()); 
String myLongitude = Double.toString(myLocation. getLongitude( )); 
myTextView = (TextView)MainActivity.this.findViewById(R. id.myTextView); 
myTextView. setText(" 当前 手机 的 位 置 如 下 :\n 经度:" 
+ myLongitude + "\n 纬度 :" + myLatitude); 
}]} 
(QOverride 
public void onProviderDisabled(String provider){ } // 若 屏蔽 提供 商 , 该 方法 被 调用 
(Override 
public void onProviderEnabled(String provider){ } // 若 激活 提供 商 , 该 方法 被 调用 
// 若 状态 发 生变 化 ,该 方法 被 调用 
(Override 
public void onStatusChanged( String provider, int status, Bundle extras) { } 
} 
private boolean checkgps( ){ 
boolean providerEnabled 
= myLocationManager. isProviderEnabled(LocationManager.GPS PROVIDER); 
if(providerEnabled == true){ // 若 被 激活 , 则 返回 真 值 
Toast. makeText(this，"GPS 模块 活动 正常 "，Toast.LENGTH SHORT) . show( ); 
return true; 
} 
elsel{ 
Toast. makeText(this, "请 开启 GPS"， Toast. LENGTH_SHORT). show( ); 
return false; 
Fy} 


上 面 这 段 代 码 在 MyCode\ MySample318\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,onLocationChanged(Location myLocation) 用 于 响应 手机 
位 置 的 改变 ,并 通过 myLocation 参数 传递 当前 的 经 度 和 纬度 值 。 另 外 ,监听 位 置 变化 在 
AndroidManifest. xml 文件 中 添加 < uses-permission android:name 一 "androlid. permission. ACCESS _ 
FINE_LOCATION "/ > 权限 和 < uses-permission android: name 王 "androld. permission。 ACCESS _ 
LOCATION_EXTRA_COMMANDS"/ > 权限。 此 实例 的 完整 项 目 在 MyCode\MySample318 文件 
夹 中 。 
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255 使 用 SensorManasger 获取 传 感 如 信息 


此 实例 主要 通过 使 用 getSystemService() 和 getSensorList() 方 法 ,实现 获取 手机 的 传感器 信息 。 
当 实 例 运 行 之 后 , 单 击 “ 获 取 手 机 的 传感器 信息 ”按钮 , 则 在 下 面 显示 当前 手机 的 所 有 传感器 信息 ,如 
图 255. 1 的 左 图 和 右 图 所 示 。 
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获取 手机 的 传感器 信息 获取 手机 的 传感器 信息 


双 检 测 该 手机 有 9 个 传感器 ， 他 们 分 别 是 : 
加 速度 传感器 accelerometer 
设备 名 称 : ACCELEROMETER 
设备 版 本 : 3 

供应 商 : MTK 
Bb 距离 传感器 proximity 
设备 名 称 : PROXIMITY 
设备 版 本 : 1 

供应 商 : MTK 
环境 光线 传感器 light 

设备 名 称 ; LIGHT 

设备 版 本 : 1 

供应 商 : MTK 
B 方向 传感器 orientation 

设备 名 称 : ORIENTATION 
设备 版 本 : 3 

供应 商 : MTK 

电磁 场 传感器 magnetic field 
设备 名 称 : MAGNETOMETER 
设备 版 本 : 3 

供应 商 : MTK 

陀螺 仪 传感器 gyroscope 
设备 名 称 : Software Gyroscope 
设备 版 本 : 3 


设备 名 称 : ROTATION VECTOR sensor 
设备 版 本 : 3 

供应 商 : Software 

Bb 未 知 传感器 

设备 名 称 : GRAVITY sensor 


图 255.1 
主要 代码 如 下 : 


// 响 应 单 击 "获取 手机 的 传感器 信息 "按钮 

public void onClickmyBtnl (View v) { 
TextView myTextView = (TextView) findViewById(R. id.myTextView); 
myTextView. setMovementMethod( ScrollingMovementMethod. getInstance( ) ) ; 
SensorManager mySensorManager = (SensorManager) 

getSystemService(Context. SENSOR SERVICE); 

// 从 传感器 管理 器 中 获得 全 部 的 传感器 列表 
List < Sensor > mySensors = mySensorManager.getSensorList(Sensor.TYPE ALL); 
myTextView. setText(" 经 检测 该 手机 有 ”+ 


mySensors. size() + "个 传感器 ,他 们 分 别 是 :\n" ); // 显 示 有 多 少 个 传感器 
for (Sensor mySensor : mySensors) { // 显 示 每 个 传感器 的 具体 信息 


String myInfo = "\n" + " 设备 名 称 :" + mySensor. getName() + "\n" 
+ "设备 版 本 :” + mySensor. getVersion() + "\n" + "供应 商 :" 
+ mySensor. getVendor() + "\n"; 
Switch (mySensor.getType()) { 
case Sensor. TYPE ACCELEROMETER: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + ”加 速度 传感器 accelerometer" + myInfo); 
break; 
Case Sensor. TYPE GYROSCOPE: 
myTextView. setText(myTextView.getText().toString() + 
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mySensor.getType() + " 陀螺 仪 传感器 gyroscope" + myInfo); 
break; 
case Sensor. TYPE LIGHT: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + "环境 光线 传感器 light"” + myInfo); 
break; 
Case Sensor. TYPE MAGNETIC FIELD: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + "电磁场 传感器 magnetic field" + myInfo); 
break; 
case Sensor. TYPE ORIENTATION: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + " 方向 传感器 orientation" + myInfo); 
break; 
Case Sensor. TYPE PRESSURE: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + ”压力 传感器 pressure" + myInfo); 
break; 
case Sensor. TYPE PROXIMITY: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + ”距离 传感器 proximity”+ myInfo); 
break; 
case Sensor. TYPE TEMPERATURE: 
myTextView. setText(myTextView. getText().toString() + 
mySensor.getType() + "温度 传感器 temperature"” + myInfo); 
break; 
default: 
myTextView. setText (myTextView. getText().toString() + 
mySensor.getType() + ”未 知 传感器 "+ myInfo); 
break; 
Ed 
} 


上 面 这 上段 代码 在 MyCode\ MySample267\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,mySensors 三 mySensorManager. getSensorList (Sensor. 
TYPE_ALL) 用 于 获取 手机 的 所 有 传感器 。 对 于 某 个 具体 的 传感器 , 它 的 具体 信息 可 以 使 用 下 列 方 
法 进行 获取 。 

(1) getMaximumRange() 方 法 ,该 方法 用 于 获取 最 大 取 值 范围 。 

(2) getName() 方 法 ,该 方法 用 于 获取 设备 名 称 。 

(3) getPower() 方 法 ,该 方法 用 于 获取 功率 。 

(4) getResolution() 方 法 ,该 方法 用 于 获取 精度 。 

(5) getType() 方 法 ,该 方法 用 于 获取 传 感 右 类 型 。 

(6) getVentor() 方 法 ,该 方法 用 于 获取 设备 供应 商 。 

(7) getVersion() 方 法 ,该 方法 用 于 获取 设备 版 本 号 。 

此 实例 的 完整 项 目 在 MyCode\MySample267 文件 夹 中 ， 


256 ”使 用 传 感 如 监测 耳 失 与 手机 听 简 的 距离 


此 实例 主要 通过 在 SensorEventListener 的 onSensorChanged() 事 件 啊 应 方法 中 监听 Proximity 
传 感 融 的 值 ,实现 监测 耳 条 与 手机 听 简 的 距离 。 当 实例 运行 之 后 ,使 用 手指 ( 耳 条 或 其 他 物体 也 一 样 ) 


<》 Anaroia 引 本 应 有 300 。 实战 篇 


在 手机 听 简 附近 任意 移动 , 当 手 指 ( 在 三 维 ) 距 离 手 机 听 简 大 约 1 厘米 左右 时 ,将 提示 距离 较 近 信息 ， 
如 图 256. 1 的 左 图 所 示 。 当 手指 超出 手机 听 简 1 厘米 范围 时 ,将 提示 距离 较 远 信息 ,如 图 256. 1 的 右 
图 所 示 。 这 主要 是 因为 Proximity 传感器 检测 物体 与 手机 的 距离 单位 是 厘米 ,一些 Proximity 传 感 需 
只 能 返回 远 和 近 两 个 状态 , 即 大 于 或 等 于 最 大 距离 返回 远 状 态 ,小 于 最 大 距离 返回 近 状 态 。Proximity 
传 感 硕 可 用 于 近 距 离 接听 电话 时 自动 关闭 LCD 屏幕 以 节省 电量 。 


MySample MySample 
当前 听 简 距离 物 ( 人 ) 体 较 近 ， 值 为 :0.0 ”| 当前 听 简 距离 物 (人 ) 体 较 远 ， 值 为 :1.0 


图 256. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements SensorEventListener { 
TextView myText; 


SensorManager mySensorManager = null; // 传感器 管理 器 
Sensor mySensor = null; // 传感器 实例 
(@Override 


protected void onCreatel( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main) ; 
myText = (TextView) findViewById(R. id. myText); 
mySensorManager = (SensorManager) getSystemService(Context.SENSOR SERVICE); 
mySensor = mySensorManager.getDefaultSensor(Sensor.TYPE PROXIMITY); 
} 
Override 
protected void onResume( ) { 
super. onResume( ) ; 
mySensorManager. registerListener(this, mySensor, 
SensorManager. SENSOR DELAY NORMAL); // 注 册 传 感 器 
} 
(Override 
protected void onPause( ) { 
super. onPause( ) ; 
mySensorManager. unregisterListener(this); // 取消 注册 传感器 
} 


(QOverride 
public void onSensorChanged( SensorEvent event) { 
float myRange = event.values[0]; 
if (myRange == mySensor. getMaximumRange()) { 
myText. setText(" 当前 听 简 距离 物 ( 人 ) 体 较 远 , 值 为 :" + myRange); 
} else { 
myText. setText(" 当前 听 简 距离 物 ( 人 ) 体 较 近 , 值 为 :”+ myRange); 
} 
myText. setTextSize( 20); 


} 
(Override 


public void onAccuracyChanged( Sensor sensor, int accuracy) { } 
} 


上 和 面 这 上段 代码 在 MyCode\ MySample271 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 其 中 ,mySensor = 二 mySensorManager. getDefaultSensor (Sensor. TYPE 
_PROXIMITY ) 表示 当前 获取 的 传感器 是 Proximity 传 感 占 ,该 传 感 融 通过 onSensorChanged 
(SensorEvent event) 的 event. values[ 0 | 获取 手机 听 简 与 物体 之 间 的 距离 。 此 外 ,监听 传感器 值 发 生 
变化 的 监听 器 必须 经 过 注册 才 有 效 , 即 : mySensorManager. registerListener (this，mySensor， 
SensorManager. SENSOR_ DELAY NORMAL) 。 此 实例 的 完整 项 目 在 MyCode\MySample271 文件 
夹 中 。 


257 ”使 用 加 速度 传 感 霸 监听 手机 的 三 维 变化 


此 实例 主要 通过 解析 SensorEventListener 的 onSensorChanged() 事 件 参 数值 SensorEvent, 实 现 
在 晃动 手机 时 监听 加 速度 传感器 的 变化 。 当 实例 运行 之 后 ,不 停 地 任意 晃动 手机 , 则 加 速度 传 感 右 的 
zx、y、z 值 将 不 断 变化 ,效果 分 别 如 图 257. 1 的 左 图 和 右 图 所 示 。 
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图 257.1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

private long lastTime = 0,curTime = 0; 

TextView myTextView; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity_main); 
myTextView = (TextView) findViewById(R. id.myTextView); 
SensorManager mySensorManager = 

(SensorManager) getSystemService(Context. SENSOR_SERVICE ) ; 
Sensor myAcceleromererSensor = // 获取 加 速度 传感器 
mySensorManager. getDefaultSensor( Sensor. TYPE ACCELEROMETER); 

SensorEventListener mySensorEventListener = new SensorEventListener() { 


(@ Override 
public void onAccuracyChanged( Sensor sensor, int accuracy) { } 
public void onSensorChanged( SensorEvent event) { // 传 感 器 数据 变动 事件 
curTime = System. currentTimeMillis( ); // 获 取 当 前 时 刻 的 ms 数 
if ((curTime — lastTime) > 100) { //100ms 检测 一 次 
// 获 取 加 速度 传感器 的 三 个 参数 
float x = event.values[SensorManager.DATA X]; 


float y = event.values[SensorManager.DATA Y]; 

float z = event.values[SensorManager.DATA 2]; 

myTextView. setText("\nX:" + x + "\nY:" + y + "\nZ:" + 2z); 
lastTime = curTime; 


| 央 疯 六 
// 在 传感器 管理 器 中 注册 监听 器 
mySensorManager. registerListener(mySensorEventListener, 
myAcceleromererSensor, SensorManager. SENSOR_ DELAY_ NORMAL); 


出 


上 面 这 段 代 码 在 MyCode\MySample269\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,x 二 event. values| SensorManager. DATA X | 、 一 event， 
values| SensorManager. DATA _Y]j 和 z=event. values| SensorManager. DATA Z| 分 别 代 表 三 维 坐 标 
的 三 个 方 回 ,它们 的 值 是 某 个 方 回 加 速度 (Acceleration) 减 去 该 方 加 的 重力 值 (Gx、Gy、Gz) ,所 以 手机 
静止 时 其 范围 是 L 一 10,10」, 例 如 ,手机 屏幕 朝 上 平 放 , 则 z==0、y 二 0、z 三 10; 这 是 因为 手机 静止 不 动 
时 ,所 有 方 回 都 没有 加 速度 , 平 放 则 产生 了 回 下 的 重力 加 速度 , 即 > 一 一 10, 因 为 重力 方 回 与 z 轴 正 问 
相反 ,所 以 相 减 后 是 10。 如 果 将 手机 回 左 倾斜 , 则 工 轴 为 正 值 。 如 果 将 手机 回 右倾 斜 , 则 z 轴 为 负 值 。 
如 果 将 手机 同上 倾斜 , 则 > 轴 为 负 值 。 如 果 将 手机 回 下 倾斜 , 则 > 轴 为 正 值 。 

此 外 ,监听 器 必须 使 用 registerListener() 方 法 注册 才 有 效 ,registerListener() 方 法 的 第 一 个 参数 
表示 监听 Sensor 事件 ,第 二 个 参数 是 Sensor 目标 传感器 类 型 ,第 三 个 参数 是 延迟 时 间 的 精度 。 延 迟 
时 间 参 数 如 表 257-1 所 示 。 

表 257-1 延迟 时 间 参 数 


参 数 延迟 时 间 (ms) 
SensorManager. SENSOR_DELAY_FASTEST 0 
SensorManager. SENSOR_DELAY GAME 20 
SensorManager. SENSOR_DELAY_UI 60 


SensorManager. SENSOR_DELAY _ NORMAL 200 
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由 于 传感器 Sensor 服务 是 否 频繁 和 快慢 都 与 电量 的 消耗 有 关 , 同 时 也 会 影响 处 理 的 效率 ,所 以 
需要 兼顾 到 消耗 电池 和 处 理 效率 的 平衡 。 此 实例 的 完整 项 目 在 MyCode\MySample269 文件 夹 中 。 


258 ”通过 传 感 如 实现 自动 进行 横 屏 和 竖 屏 切换 


此 实例 主要 通过 使 用 自 定义 类 MyUtils 监听 传感器 的 变化 ,实现 横 屏 和 坚 屏 自动 切换 。 当 实例 
运行 之 后 ,在 “网 址 : ”输入 框 中 输入 一 个 网 址 ,如 “http://www. sohu. com/”, 单 击 “ 浏 览 ” 按 钮 , 则 在 
WebView 控件 中 显示 该 网 页 ,如 果 竖 屏 放 置 手机 , 则 此 应 用 的 竖 屏 显示 效果 如 图 258. 1 的 左 图 所 示 ; 
如 果 横 屏 放 置 手 机 , 则 此 应 用 的 横 屏 显示 效果 如 图 258. 1 的 右 图 所 示 ; 即 此 应 用 能 够 通过 传感器 的 变 
化 自动 决定 是 以 横 屏 显示 或 是 以 竖 屏 显示 。 

MySample 
网 址 : http://www.sohu.com/ 


浏览 


/Woonyos mmm//:dny :开罗 


< 搜狐 首页 ”娱乐 


明星 库 电影 电视 音乐 
独家 ” 韩 娱 日 娱 。 ”欧美 专题 


有 新 男友 ?张柏芝 2018 要 生 宝 宝 


张柏芝 耗 四 小 时 献血 超 满足 因 太 瘦 险 被 拒 
张柏芝 带 儿 子 去 游乐 园 Lucas 闫 有 谢霆锋 风采 


要 闻 排行 独家 .八卦 | 


图 258.1 


主要 代码 如 下 : 


Override 

protected void onPause() { // 停 止 监听 
super. onPause( ) ; 
MyUtils. getInstance(this). stop(); 

} 

(Override 

protected void onResume() { // 开 始 监听 
super. onResume( ) ; 
MYUtils. getInstance(this). start(this); 
onClickmyBtnl(nul1l) ; 

} 

public void onClickmyBtnl (View v) { // 啊 应 单 击 "浏览 "按钮 
myWebView. loadUrl (myEditText. getText().toString( )); 


} 


上 面 这 段 代 码 在 MyCode\ MySample762\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,MyUtils. getInstance(this). start(this) 表 示 启 用 手机 传 感 
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船 的 监听 功能 。MyUtils 是 一 个 自 定 义 类 ,该 类 的 主要 代码 如 下 : 


public class MyUtils { 
private static MyUtils myInstance; 
private Activity myActivity; 
private boolean isLandscape = false; // 上 默认 是 竖 屏 
private SensorManager mySensorManager:; 
private OrientationSensorListener myOrientationSensorListener; 
private Sensor mySensor; 


// 根 据 重 力 感应 监听 结果 ,改变 屏幕 朝向 
private Handler myHandler = new Handler(Looper. getMainLooper()) { 
public void handleMessage(Message msg) { 
if (msg.what == 888) { 
int orientation = msg.argl; 
// 根 据 手机 屏幕 的 朝向 角度 ,设置 内 容 的 横竖 屏 , 并 且 记 录 状 态 
if (orientation > 45 orientation < 135) { 
myActivity. setRequestedOrientation( 
ActivityInfo. SCREEN ORIENTATION REVERSE LANDSCAPE); 
isLandscape = true; 
} else if (orientation > 135 orientation < 225) { 
myActivity. setRequestedOrientation( 
ActivityInfo. SCREEN ORIENTATION REVERSE PORTRAIT); 
isLandscape = false; 
} else if (orientation > 225 & orientation < 315) { 
myActivity. setRequestedOrientation( 
ActivityInfo. SCREEN ORIENTATION LANDSCAPE); 
isLandscape = true; 
} else if ((orientation > 315 & orientation < 360) 
|| (orientation > 0 forientation < 45)) { 
myActivity. setRequestedOrientation( 
ActivityInfo. SCREEN ORIENTATION PORTRAIT); 
isLandscape = false; 
}) } } ); 
public static MyUtils getInstance(Context context) { // 获 取 实 例 对 象 
if (myInstance == null) { 
synchronized (MyUtils.class) { 
if (myInstance == null) { myInstance = new MyUtils(context); } 
1 
return myInstance; 
} 
private MyUtils(Context context) { // 初 始 化 重力 感应 器 
mySensorManager = 
(SensorManager) context. getSystemService( Context. SENSOR SERVICE); 
mySensor = mySensorManager. getDefaultSensor(Sensor.TYPE GRAVITY); 
myOrientationSensorListener = new OrientationSensorListener(myHandler); 
} 
public class OrientationSensorListener implements SensorEventListener { 
private static final int DATA X = 0; 
private static final int DATA Y = 1; 
private static final int DATA 2Z = 2; 
public static final int ORIENTATION UNKNOWN = -1; 
private Handler rotateHandler; 
public OrientationSensorListener(Handler handler){ rotateHandler = handler;} 


public void onAccuracyChanged( Sensor arg0, int argql) { } 
public void onSensorChanged( SensorEvent event) { 
float[ ] values = event.values; 
int orientation = ORIENTATION _ UNKNOWN; 
float X = - values[_DATA X]; 
float Y = -~ values[_DRTR Y]; 
float Z = - values[ DATA 2Z]; 
float magnitude = X* X+Y*xY; 
if (magnitude x 4>= Z* 2){ // 在 屏幕 旋转 时 
float OneEightyOverPi = 57.29577957855f; 
float angle = (float) Math.atan2( - Y, X) * OneEightyOverPi; 
orientation = 90 - Math. round(angle); 
while (orientation >= 360) { orientation -= 360; } 
while (orientation < 0) { orientation += 360; } 
} 
if (rotateHandler != null) { 
rotateHandler. obtainMessage(888, orientation, 0). sendToTarget( ); 
Eh 
public void start(Activity activity) { // 开 始 监听 
myActivity = activity; 
mySensorManager. registerListener(myOrientationSensorListener, 
mySensor, SensorManager. SENSOR DELAY UI); 


} 
public void stop() { // 停 止 监听 
mySensorManager. unregisterListener(myOrientationSensorListener); 
myActivity = null; // 防 止 内 存 泄漏 
而 
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上 面 这 段 代 码 在 MyCode\MySample762\app\src\main\java\com\bin\luo\mysample\ MyUtils. 


259 ”使 用 setRequestedOrientation( ) 实 现 横 屏 


java 文件 中 。 需 要 说 明 的 是 ,访问 网 络 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission 
android: name 三 " android. permission. INTERNET"/ > 权限 。 此 实例 的 完整 项 目 在 MyCodeA\ 
MySample762 文件 夹 中 。 


此 实例 主要 通过 在 setRequestedOrientation ( ) 方法 中 设置 ActivityInfo. SCREEN _ 


的 右 图 所 示 。 
主要 代码 如 下 : 
public void onClickmyBtnl(View v) { // 响 应 单 击 " 横 屏 显示 "按钮 
setRequestedOrientation(ActivityInfo. SCREEN ORIENTATION LANDSCAPE); 
} 
public void onClickmyBtn2(View v) { // 响 应 单 击 " 竖 屏 显示 "按钮 


setRequestedOrientation(ActivityInfo. SCREEN ORIENTATION PORTRAIT); 
} 


ORIENTATION_LANDSCAPE 参数 或 ActivityInfo. SCREEN_ORIENTATION _PORTRAIT 参 
数 ,实现 限制 应 用 屏幕 始终 以 横 屏 或 竖 屏 显示 。 当 实例 运行 之 后 , 单 击 “ 竖 屏 显示 ”按钮 , 则 应 用 屏幕 
将 以 竖 屏 显示 ,如 图 259. 1 的 左 图 所 示 。 单 击 “ 横 屏 显示 ”按钮 , 则 应 用 屏幕 将 以 横 屏 显示 ,如 图 259. 1 


上 F 面 这 段 代 码 在 MyCode\ MySample235\app\src\ main\java\com\bin\luo\mysample\ 
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图 259.1 


MainActivity. java 文件 中 。 在 这 段 代 码 中 , setRequestedOrientation ( ActivityInfo. SCREEN _ 
ORIENTATION _LANDSCAPE ) 用 于 横 屏 显示 ,setRequestedOrientation (ActivityInfo. SCREEN _ 
ORIENTATION_PORTRAIT) 用 于 坚 屏 显示 。 实 际 上 ,在 许多 Android 应 用 中 ,为 了 避免 横竖 屏 切 
换 引 发 的 不 必要 膝 烦 ,通常 禁止 默认 的 横竖 屏 切 换 , 即 在 AndroidManifest. xml 中 通过 设置 android: 
screenOrientation 属性 值 ,android:screenOrientation 属性 有 以 下 几 个 属性 值 。 

(1) unspecified ,默认 值 , 由 系统 来 判断 显示 方向 ,判定 的 策略 是 和 设备 相关 的 ,所 以 不 同 的 设备 
会 有 不 同 的 显示 方 回 。 

(2) landscape, 檬 屏 显 示 , 宽 比 高 长 。 

(3) portrait , 坚 屏 显示 ,高 比 宽 长 。 

(4) user, 用 户 当 前 首选 的 方 回 。 

(5) behind, 和 该 Activity 下 面 的 那个 Activity 的 方 回 一 致 (在 Activity 堆栈 中 ) 。 

(6) sensor, 由 物理 感应 需 决 定 , 如 果 用 户 旋转 设备 , 则 屏幕 会 进行 横竖 屏 切 换 。 

(7) nosensor, 忽 略 物 理 感应 髓 ,这 样 就 不 会 随 着 用 户 旋转 设备 而 更 改 了 (unspecified 设置 除外 ) 。 

比如 ,如 果 设 置 androlid: screenOrientation 王 "portralt", 则 无 论 手 机 如 何 变动 ,拥有 这 个 属性 的 
Activity 都 竖 屏 显示 ; 如 果 设 置 android:screenOrientation 王 “landscape", 则 横 屏 显示 。 此 实例 的 完 
整 项 目 在 MyCode\MySample235 文件 夹 中 ， 


260 ”根据 手机 是 模 屏 或 是 竖 拼 进行 控件 布局 


此 实例 主要 通过 监听 onConfigurationChanged() 事 件 啊 应 方法 ,实现 根据 手机 是 横 屏 或 是 竖 屏 进 
行 控件 布局 。 当 实例 运行 之 后 ,如 果 当 前 手机 处 于 竖 屏 状态 , 则 将 同时 显示 两 幅 图 像 ,如 图 260.1 的 
左 图 所 示 。 如 果 当 前 手机 处 于 横 屏 状态 , 则 仅 显 示 下 面 的 图 像 ,如 图 260. 1 的 右 图 所 示 。 注 意 : 如 果 
测试 不 成 功 , 请 将 手机 设置 中 的 “ 竖 屏 ” 改 为 “自动 旋转 ”。 
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图 260. 1 
主要 代码 如 下 : 


public void onConfigurationChanged(Configuration myConfig) { 
super. onConf igurationChanged( myConfig); 
ViewGroup. LayoutParams myParams = myImageView2. getLayoutParams( ); 
if(myConfig. orientation == Configuration.ORIENTATION PORTRAIT){ 


myImageView1. setVisibility(View. VISIBLE); // 当 坚 屏 时 ,显示 两 幅 图 像 
}else if(myConfig. orientation == Configuration.ORIENTATION LANDSCAPE){ 

myImageView1. setVisibility(View. GONE); // 当 横 屏 时 , 仅 显 示 下 面 的 图 像 
} 
myImageView2. setLayoutParams (myParams ) ; 


上 F 面 这 段 代 码 在 MyCode\ MySample316 \app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中, if (myConfig. orientation 三 三 Configuration. 
ORIENTATION_PORTRAIT) 表 示 当 前 手机 是 竖 屏 ,if(myConfig. orientation 一 一 Configuration. 
ORIENTATION_LANDSCAPE) 表示 当前 手机 是 横 屏 ,myImageView]l. setVisibility( View. GONE) 
表示 暂时 不 显示 myImageViewl 控件 。 另 外 ,监听 onConfigurationChanged ( ) 需 要 首先 在 
AndroidManifest. xml 文件 中 修改 配置 < activity android: name 三 ". MainActivity ” android : 
configChanges 一 “orientation | screenSize "> 和 添加 权限 < uses-permission android: name 二 "android. 


permission. CHANGE_CONFIGURATION "/ >。 此 实例 的 完整 项 目 在 MyCode\MySample316 文件 夹 中 。 


261 使 用 FLAG FULLSCREEN 标志 实现 全 屏 显 示 


此 实例 主要 通过 使 用 getWindow() 的 setFlags() 方 法 ,实现 控制 应 用 的 界面 是 否 以 全 屏 模式 显 
示 。 当 实例 运行 之 后 , 单 击 “ 全 屏 显 示 ?按钮 , 则 应 用 界面 的 标题 栏 将 隐藏 ,如 图 261. 1 的 左 图 所 示 。 


JS Android 炫 酷 应 用 300 例 ， 实战 篇 


单 击 “ 退 出 全 屏 ” 按 钮 , 则 将 恢复 默认 界面 ,显示 标题 栏 ,如 图 261. 1 的 右 图 所 示 。 


MySample 


public class MainActivity extends Activity { 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
boolean isNoTitle = getIntent().getBooleanExtra("isNoTitle", false); 
requestWindowFeature(isNoTitle? 
Window. FEATURE NO _ TITLE: Window. FEATURE ACTION BAR); 
getWindow( ) . setFlags(WindowManager. LayoutParams. FLAG FULLSCREEN, 
WindowManager. LayoutParams. FLAG FULLSCREEN); // 设 置 全 屏 

setContentView(R. layout.activity main); 

} 
// 响 应 单 击 "全 屏 显 示 " 按 钮 

public void onClickmyBtnl (View v) { 
Intent myIntent = getIntent(); 
myIntent. putExtra("isNoTitle", true); 
finish( ); 
startActivity(myIntent); 

} 
// 响 应 单 击 " 退 出 全 屏 " 按 钮 

public void onClickmyBtn2(View v) { 
Intent myIntent = getIntent(); 
myIntent. putExtra("isNoTitle", false); 
finish( ) ; 
startActivity(myIntent); 

上 


上 面 这 段 代 码 在 MyCode\ MySample247\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 上 段 代 码 中 ，getWindow ( ). setFlags (WindowManager. 
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LayoutParams. FLAG _ FULLSCREEN, WindowManager. LayoutParams. FLAG _FULLSCREEN) 用 
于 全 屏 显 示 应 用 。 此 实例 的 完整 项 目 在 MyCode\MySample247 文件 夹 中 。 


262 ”使 用 Display 获取 屏幕 宽度 和 高 度 


此 实例 主要 通过 使 用 Display 的 getWidth() 方 法 和 getHeight() 方 法 ,获取 当前 手机 的 屏幕 宽度 
和 屏幕 高 度 。 当 实例 运行 之 后 , 单 击 “ 获 取 屏 幕 的 宽度 和 高 度 ” 按 钮 , 则 在 弹出 的 Toast 中 显示 当前 手 
机 的 屏幕 宽度 和 屏幕 高 度 ,如 图 262. 1 所 示 。 


MySample 


获取 屏幕 的 宽度 和 高 度 


主要 代码 如 下 : 


// 响 应 单 击 "获取 屏幕 的 宽度 和 高 度 "按钮 
public void onClickmyBtnDiaplay(View v) { 


Display myDisplay = getWindowManager().getDefaultDisplay( ); 
int myWidth = myDisplay. getWidth( ); // 获 取 屏 幕 宽 度 
int myHeight = myDisplay.getHeight(); // 获 取 屏 幕 高 度 


Toast. makeText (getApplicationContext( )," 屏幕 宽度 :”+ myWidth 
+ " ,屏幕 高 度 :" + myHeight, Toast.LENGTH SHORT) . show(); 


上 面 这 上段 代码 在 MyCode\ MySample078\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample078 文件 夹 中 。 


263 ”使 用 StatFs 获取 内 部 总 空间 和 可 用 衬 间 大 小 


此 实例 主要 通过 使 用 StatFs 的 getBlockCount() 方 法 、getAvailableBlocks() 方 法 、getBlockSize() 
方法 和 Environment 的 getDataDirectory() 方 法 ,从 而 获取 手机 内 部 的 总 空间 和 可 用 空间 大 小 。 当 实 


CD》 Anaroia 红 本 应 局 300 。 实战 篇 


例 运 行 之 后 , 单 击 “ 获 取 手 机 内 部 总 空间 大 小 ”按钮 , 则 在 弹出 的 Toast 中 显示 手机 内 部 的 总 空间 大 
小 ,如 图 263. 1 的 左 图 所 示 ; 单 击 “ 获 取 手 机 内 部 可 用 空间 大 小 ”按钮 , 则 在 弹出 的 Toast 中 显示 手机 
内 部 的 可 用 空间 大 小 ,如 图 263. 1 的 右 图 所 示 。 


MySample MySample 


获取 手机 内 部 总 空间 大 小 获取 手机 内 部 总 空间 大 小 
获取 手机 内 部 可 用 空间 大 小 获取 手机 内 部 可 用 空间 大 小 


于 机 内 部 总 空间 大 小 是 : 11.89 GB 手机 内 部 可 用 空间 大 小 是 : 1.34 GB 


图 263.1 


// 响 应 单 击 "获取 手机 内 部 总 空间 大 小 "按钮 
public void onClickmyBtnl(View v) { 
File myPath = Environment.getDataDirectory(); 
StatFs myStatFs = new StatFs(myPath. getPath( )); 
long myBlockSize = myStatFs.getBlockSizel( ); 
long myBlockCount = myStatFs.getBlockCount( ); 
long myTotalSize = myBlockCount x* myBlockSize; 
String mySize = Formatter. formatFileSize(this, myTotalSize); 
Toast. makeText(this, "手机 内 部 总 空间 大 小 是 :" 
+ mySize, Toast. LENGTH SHORT). show( ); 
} 
// 响 应 单 击 " 获 取 手 机 内 部 可 用 空间 大 小 "按钮 
public void onClickmyBtn2(View v) { 
File myPath = Environment.getDataDirectory(); 
StatFs myStatFs = new StatFs(myPath. getPath( )); 
long myBlockSize = myStatFs.getBlockSizel( ); 
long myAvailableBlocks = myStatFs.getAvailableBlocks( ); 
long myAvailableSize = myAvailableBlocks * myBlockSize; 
String mySize = Formatter.formatFileSize(this, myAvailableSize); 
Toast. makeText(this, "手机 内 部 可 用 空间 大 小 是 :" 
+ mySize, Toast.LENGTH SHORT). show( ) ; 


上 面 这 段 代码 在 MyCode\ MySample667\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 上 面 这 段 代 码 中 ,myBlockSize = 二 myStatFs. getBlockSize() 用 于 获取 
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单个 扇 区 的 大 小 。myBlockCount = 二 myStatFs.getBlockCount ( ) 用 于 获取 全 部 扇 区 的 数量 。 
myAvailableBlocks 二 myStatFs. getAvailableBlocks() 用 于 获取 可 用 扇 区 的 数量 。 然 后 恒 区 大 小 * 
扇 区 数量 即 为 空间 大 小 。myPath 二 Environment. getDataDirectory() 用 于 获取 根 目 录 /data 内 部 存 
储 路 径 。Environment 获取 其 他 目录 的 方法 还 有 。 

(1) Environment. getDownloadCacheDirectory() ,该 方法 用 于 获取 缓存 目录 /cache。 

(2) Environment. getExternalStorageDirectory() ,该 方法 用 于 获取 SD 卡 目录 /mnt/ sdcard, 即 
手机 外 置 sd 卡 的 路 径 。 

(3) Environment. getRootDirectory() ,该 方法 用 于 获取 系统 目录 /system 。 

此 实例 的 完整 项 目 在 MyCode\MySample667 文件 夹 中 。 


264 ”使 用 GestureDetector 实现 纵 困 滑 动 切 换 


此 实例 主要 通过 使 用 GestureDetector 动态 监听 控件 操作 ,并 重 写 onFling() 方 法 监听 滑动 事件 ， 
从 而 啊 应 在 ViewFlipper 控件 中 上 下 滑动 时 切换 图 像 。 当 实例 运行 之 后 , 回 上 滑动 手指 , 则 
ViewFlipper 控件 滑 人 下 一 幅 图 像 ,向 下 滑动 手指 , 则 ViewFlipper 控件 滑 和 人 上 一 幅 图 像 ,效果 分 别 如 
图 264. 1 的 左 图 和 右 图 所 示 。 


区 bd 
t ? 


MySample 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
int[ ] myImages = {R.mipmap.myimagel, R.mipmap.myimage2, 
R. mipmap. myimage3，R. mipmap. myimage4，R.mipmap.myimage5 } ; 
ViewFlipper myViewF] ipper; 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 


setContentView(R. layout. activity main); 
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myViewFlipper = (ViewFlipper) findViewById(R. id.myViewFlipper); 
for (int i = 0; i< myImages. length; i++) { 
ImageView myImageView = new ImageView(MainActivity. this); 
myImageView. setImageResource(myImages[ i]); 
myImageView. setScaleTypel( ImageView. ScaleType. FIT XY); 
myViewFlipper.addView(myImageView); 
} 
myViewFlipper. setClickable(true); // 设 置 ViewFlipper 控件 可 点 击 
final GestureDetector myGestureDetector = 
new GestureDetector(new GestureDetector. SimpleOnGestureListener() { 
(@ Override 
// 检 测 手 势 滑动 事件 
public boolean onFling(MotionEvent el, MotionEvent e2, 
float velocityX, float velocityY) { 
if (e2.getY() > el.getY()) showNextView( ); // 向 下 滑动 
else if (el.getY() > e2.getY()) showPreviousView();  // 向 上 滑动 
return false; 
} }); 
myViewFlipper. setOnTouchListener(new View. OnTouchListener() { 
(@ Override 
public boolean onTouch(View v, MotionEvent event) { 
// 传 递 事 件 至 手势 识别 器 对 象 
return myGestureDetector. onTouchEvent (event ); 
} }); 
} 
public void showPreviousView() { 
// 设 置 下 滑 时 ,ViewFlipper 子 视图 进 场 和 退场 动画 
myViewFlipper. setInAnimation(AnimationUtils. loadAnimation(this, 
R.anim. slide bottom in)); 
myViewFlipper. setOutAnimation(AnimationUtils. loadAnimation(this, 
R.aninm. slide top out)); 
myViewFlipper. showPrevious( ); // 显 示 上 一 幅 图 像 
} 
public void showNextView() { 
// 设 置 上 滑 时 ,ViewFlipper 子 视图 进 场 和 退场 动画 
myViewFlipper. setInAnimation(AnimationUtils. loadAnimation(this, 
R. anim. slide top_ in)); 
myViewFlipper. setOutAnimation( AnimationUtils. loadAnimation(this, 
R.anim. slide bottom out)); 
myViewFlipper. showNext( ); // 显 示 下 一 幅 图 像 
} 
} 


上 面 这 段 代 码 在 MyCode\ MySample835\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 , myViewFlipper. setInAnimation (AnimationUltils. 
loadAnimation(this，R. anim. slide_bottom_in)) 用 于 设置 在 手指 上 滑 时 图 像 的 进 场 动画 ,关于 R. 
anim. slide_bottom_in 进 场 动画 的 详细 内 容 请 参考 源 代码 中 的 MyCode\MySample835\app\src\main 
\res\anim\slide_bottom in. xml 文件 。 此 实例 的 完整 项 目 在 MyCode\MySample835 文件 夹 中 ， 
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265 自 定 义 手 机 振动 器 (Vibrator) 的 振动 模式 


此 实例 主要 通过 设置 Vibrator 的 vibrate() 方 法 的 模式 参数 ,实现 自 定义 手机 振动 器 (Vibrator) 
的 振动 模式 。 当 实例 运行 之 后 , 单 击 “ 自 定义 振动 模式 1” 按 钮 , 则 手机 将 按照 指定 的 模式 1 进行 振动 ; 
单 击 “ 自 定义 振动 模式 2” 按 钮 , 则 手机 将 按照 指定 的 模式 2 进行 振动 ,效果 分 别 如 图 265. 1 的 左 图 和 
右 图 所 示 。 


MySample MySample 


自 定义 振动 模式 1 ” 自 定义 振动 模式 2 || 自 定义 振动 模式 1 ” 自 定义 振动 模式 2 


正在 根据 自 定 义 模式 1 进行 振动 正在 根据 自 定 义 模式 2 进行 振动 


图 265. 1 
主要 代码 如 下 : 


// 响 应 单 击 " 自 定义 振动 模式 1" 按 钮 
public void onClickmyBtnl (View v) { 
Vibrator myVibrator = (Vibrator) getSystemService(VIBRATOR SERVICE); 
long[ ] myPattern = {3000, 1000, 2000, 5000, 3000, 1000}; 
myVibrator. vibrate(myPattern, - 1); 
Toast. makeText (this, 
"正在 根据 自 定义 模式 1 进行 振动 "，Toast. LENGTH_SHORT). show( ) ; 
} 
// 响 应 单 击 按钮 " 自 定义 振动 模式 2" 
public void onClickmyBtn2(View v) { 
Vibrator myVibrator = (Vibrator) getSystemService(VIBRATOR SERVICE); 
long[ ] myPattern = {300, 100, 200, 500, 300, 100}; 
myVibrator. vibrate(myPattern, - 1); 
Toast. makeText (this, 
"正在 根据 自 定义 模式 2 进行 振动 "，Toast. LENGTH_SHORT). show( ); 


上 面 这 段 代 码 在 MyCode\ MySample370\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 在 这 段 代码 中 ,vibrate() 方 法 如 果 只 有 1 个 参数 , 则 该 参数 用 来 指定 振动 
的 ms 数 。 如 果 该 方法 有 两 个 参数 , 则 第 1 个 参数 用 来 指定 振动 的 模式 ,第 2 个 参数 用 来 指定 是 否 需 
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要 循环 ,如 vibrate(myPattern， 一 1) 中 的 myPattern 即 是 预 置 的 振动 模式 ,myPattern 二 {3000， 
1000，2000, 5000,， 3000,，1000} 表 示 等 待 3 秒 后 ,振动 1 秒 , 再 等 待 2 秒 后 ,振动 5 秒 ,再 等 待 3 秒 后 ， 
振动 1 秒 。 此 外 ,操作 振动 器 需要 在 AndroidManifest. xml 文件 中 添加 < uses-permission android : 
name 一 "android. permission. YIBRATE"/ 作 权限。 此 实例 的 完整 项 日 在 MyCode\MySample370 文件 
夹 中 。 


266 ”使 用 SurfaceView 实现 照相 机 的 预览 功能 


此 实例 主要 实现 了 使 用 SurfaceView 控件 实现 照相 机 的 预览 功能 。 当 实例 运行 之 后 ,摄像 头 获 
取 的 预览 图 像 和 直接 通过 SurfaceView 控件 显示 出 来 ,如 图 266. 1 的 左 图 所 示 ; 单 击 “开始 照相 ”按钮 ， 
则 当前 显示 的 图 像 将 以 照片 的 形式 保存 在 “手机 存储 ”目录 中 ,如 图 266. 1 的 右 图 所 示 。 
二 ”| 愉 62% 下 午 7:30 ] 二 ”4 证 62% 下 午 7:31 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 

Camera myCamera; 

SurfaceView mySurfaceView; 

SurfaceHolder myHolder; 

(Override 

Protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
mySurfaceView = (SurfaceView) findViewById(R. id.mySurfaceView); 
myHolder = mySurfaceView. getHolder(); 
myHolder. setFixedSize(176, 155); 
myHolder. setKeepScreenOn(true); 
myHolder. setType( SurfaceHolder. SURFACE TYPE PUSH BUFFERS); 
myHolder.addCallback(new SurfaceHolder. Callback() { 
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(Override 
public void surfaceCreated(SurfaceHolder holder) { 
try { 
myCamera = Camera.open( ); 
if (myCamera == null) { 
int myCameraCount = Camera. getNumberOfCameras( ); 
myCamera = Camera.open(myCameraCount - 1); 


} 

Camera. Parameters myParams = myCamera.getParameters( ); 

myParams. setJpegQuality(80); // 设 置 照片 质量 

myParams. setPictureSize(1024, 768); // 设 置 照片 大 小 

myParams. setPreviewFrameRate(5); // 预 览 帧 率 

myCamera. setDisplayOrientation(90); // 默 认为 横 屏 ,旋转 90 度 至 竖 屏 


myCamera. setPreviewDisplay(holder); 
myCamera. setPreviewCallback(new Camera. PreviewCallback() { 
@Override 
public void onPreviewFrame(byte[ ] data, Camera camera) { } 
}); 
myCamera. startPreview( ); 
} catch (Exception e) { e.printStackTrace( ); } 
} 
(@ Override 
public void surfaceChanged!( SurfaceHolder holder, 
int format, int width, int height) { } 
(Override 
public void surfaceDestroyed( SurfaceHolder holder) { 
holder. removeCallback(this); 
myCamera. setPreviewCallback (null); 
myCamera. stopPreview( ); 
myCamera. lock( ); 
myCamera. releasel( ); 
myCamera = null; 
ist 
public void onClickButton(View v) { // 响 应 单 击 " 开 始 照 相 " 按 钮 
myCamera. takePicture(null, null, new Camera.PictureCallback() { 
(Override 
public void onPictureTaken(byte[ ] data, Camera camera) { 
String myFileName = 
new SimpleDateFormat("yyyy ~ MM—- dd— HH mm— ss").format(new Date( )); 
File myFile = new File(Environment. getExternalStorageDirectory!() 
+ "/" + myFileName + ".png"); 
try { 
File0utputStream myStream = new FileOQutputStream(myFile); 
myStream. writel( data); 
myStream. close( ) ; 
Toast. makeText(MainActivity. this, 
myFile.getPath() + "照相 文件 保存 成 功 !", Toast. LENGTH_LONG). show( ) ; 
camera. startPreview( ); // 重 新 开始 预览 
} catch (Exception e) { e. PrintStackTrace(); } } }); 
} 
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上 F 面 这 段 代 码 在 MyCode\ MySample678\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,照相 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission android: name = " android. permission. CAMERA "/> 和 < uses-permission android: 


name 一 "android. permission. WRITE_EXTERNAL_STORAGE"/> 权 限 。 此 实例 的 完整 项 目 在 
MyCode\MySample678 文件 夹 中 ，。 


267 ”使 用 Camera 实现 缩小 和 放大 预 宠 夯 面 


此 实例 主要 通过 使 用 setZoom() 方 法 控制 照相 机 参数 Camera. Parameters ,实现 在 摄像 头 预 览 时 
缩小 和 放大 预览 画面 。 当 实例 运行 之 后 ,将 自动 打开 摄像 头 的 预览 画面 , 单 击 * 放 大 当前 视角 ?按钮 ， 
则 预览 画面 放大 ,如 图 267. 1 的 左 图 所 示 ; 单 击 “缩小 当前 视角 ?按钮 , 则 预览 画面 缩小 ,如 图 267. 1 的 
右 图 所 示 。 


= De WY 


MySample MySample 


放大 当前 视角 缩小 当前 视角 放大 当前 视角 缩小 当前 视角 


图 267. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements SurfaceHolder. Callback { 
Camera myCamera; 
int myZoom = 5; 
private SurfaceHolder myHolder; 
SurfaceView mySurfaceView; // 用 于 显示 预览 画面 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
setContentView(R. layout.activity main); 
mySurfaceView = (SurfaceView) findViewById(R. id.mySufaceView); 
myHolder = mySurfaceView. getHolder( ); 
myHolder. addCallback(this); 
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(Override 
public void surfaceCreated(SurfaceHolder holder) { 
myCamera = Camera. open( ) ; 


try { 
myCamera. setDisplayOrientation(90); 
myCamera. setPreviewDisplay(holder); // 通 过 surfaceview 显示 预览 画面 
myCamera. startPreview( ) ; // 开 始 预览 


myCamera. getParameters( ) . setZoom(myZoom) ; 
} catch ( IOException e) { e. printStackTrace(); } 
} 
(Override 
public void surfaceChanged(SurfaceHolder holder， 
int format, int width，int height){ } 
(Override 
public void surfaceDestroyed( SurfaceHolder holder) { 
myCamera. stopPreview( ); 
myCamera. release( ); 
myCamera = null; 
} 
public void onClickBtnl (View v) { // 响 应 单 击 "放大 当前 视角 "按钮 
if (myCamera. getParameters( ) . isZoomSupported( ) ) { 
Camera. Parameters myParams = myCamera. getParameters(); 
myParams. SetZoom(myZoom += 1); 
myCamera. setParameters(myParams); 
遍 ， 
public void onClickBtn2(View v) { // 响 应 单 击 "缩小 当前 视角 "按钮 
if (myCamera. getParameters(). isZoomSupported( ) ) { 
Camera. Parameters myParams = myCamera. getParameters( ); 
myParams. setZoom(myZoom -= 1); 
myCamera. SetParameters(myParams ) ; 
Fy 


上 面 这 段 代 码 在 MyCode\ MySample764\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myParams 二 myCamera. getParameters() 用 于 获取 照相 
机 参数 , myParams. setZoom (myZoom 一 = 1) 用 于 设置 照相 机 的 缩放 系数 , myCamera. 
setParameters(myParams) 用 于 设置 照相 机 的 参数 。 此 外 ,操作 照相 机 需要 在 AndroidManifest. xml 
文件 中 添加 < uses-permission android:name 一 "android. permission. CAMERA"/ > 权限 。 此 实例 的 完 
整 项 目 在 MyCode\MySample764 文件 夹 中 ， 


268 ”使 用 Camera 实现 预览 时 摄像 头 手 动 对 焦 


此 实例 主要 通过 使 用 SurfaceView 控件 预览 摄像 头 男 面 ,并 通过 Camera 对 象 的 方法 
onAutoFocus() 获 取 对 焦 结果 ,从 而 实现 为 摄像 头 预 览 添 加 手动 对 焦 功 能 。 当 实例 运行 之 后 ,将 自动 
打开 摄像 头 的 预览 画面 ; 如 果 某 处 画面 模糊 不 清 , 点 击 模糊 不 清 区 域 则 重新 以 该 处 为 焦点 进行 对 焦 操 
作 ( 出 现 红色 的 方 框 ); 当 对 焦 操 作 完 成 时 ,红色 对 焦 方 框 自 动 消失 ,该 区 域 画面 会 逐渐 变 得 清晰 , 且 在 
屏幕 下 方 以 Toast 形式 提示 对 焦 结 果 , 如 图 268.1 所 示 。 测 试 时 , 单 击 预览 画面 的 非 中 心 位 置 , 当 这 
些 地 方 从 非 焦点 变 成 焦点 之 后 ,画面 立刻 变 得 清晰 起 来 。 
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MySample 


手动 对 焦 成 功 ! 


268. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements SurfaceHolder. Callback, View. OnClickListener, 
View. OnTouchListener { 
Camera myCamera; 
private SurfaceView mySurfaceView; 
private SurfaceHolder myHolder; 
private ImageView myRectView; // 对 焦 方 框 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity main); 
mySurfaceView = (SurfaceView) findViewById(R. id.MySurfaceView); 
myHolder = mySurfaceView. getHolder( ); 
myHolder. addCallback (this); 
mySurfaceView. setOnClickListener(this); 
mySurfaceView. setOnTouchListener(this); 
myRectView = (ImageView) findViewById(R. id.MyRectView); 
myRectView. setVisibility(View. GONE); 
} 
(Override 
public void surfaceCreated(SurfaceHolder holder) { 
if (myCamera == null) { // 开 启 相机 
myCamera = Camera. open(); 
try { 
myCamera. setDisplayOrientation(90); 
myCamera. setPreviewDisplay(holder); // 通 过 SurfaceView 预览 画面 
myCamera. startPreview( ); // 开 始 预览 
} catch (IOException e) { e. printStackTrace(); } 
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上 
(Override 
public void surfaceChanged(SurfaceHolder holder， 
int format, int width，int height){ } 
(Override 
public void surfaceDestroyed( SurfaceHolder holder) { 
myCamera. stopPreview( ) ; 
myCamera. releasel( ) ; 
myCamera = null; 
} 
(Override 
public void onClick(View v) { 
if (v.getId() == R. id.MySurfaceView) { 
if (myCamera != null) { 
myCamera. autoFocus(new Camera. AutoFocusCallback( ) { 
@Override 
public void onAutoFocus( boolean success, Camera camera) { 
if (success) { 
Toast. makeText (MainActivity. this, 
"手动 对 焦 成 功 !" ，Toast.LENGTH LONG) . show( ); 
} else { 
camera. autoFocus(this); // 如 果 失 败 , 自 动 对 焦 
Toast. makeText (MainActivity. this, 
"手动 对 焦 失 败 , 己 调整 为 自动 对 焦 !"， Toast.LENGTH _LONG). show(); 
Pal) 
(Override 
public boolean onTouch(View v，MotionEvent event){ 
if (event. getRction() == MotionEvent.ACTION DOWN) { // 绘 制 对 焦 方 框 
float x 
float y = event.getY(); 
Paint myPaint = new Paint( ); 
myPaint. setStrokeWidth(5); 
myPaint. setStylel(Paint. Style. STROKE ) ; 
myPaint. setColor(Color. RED); 
Bitmap myBitmap = Bitmap. createBitmap( 
getWindowManager( ).getDefaultDisplay().getWidth( ), 
getWindowManager( ).getDefaultDisplay( ).getHeight( ), 
Bitmap. Config. ARGB_8888); 
Canvas myCanvas = new Canvas(myBitmap); 
myCanvas. drawRect(x ~ 75,y - 75, x + 75, y + 75, myPaint); 
myRectView. setImageBitmap(myBitmap); 
myRectView. setVisibility(View. VISIBLE); 
} else if (event.getAction() == MotionEvent.ACTION UP) { 
myRectView. setVisibility(View.GONE); 
} 
return false; 


} } 


event. getX( ); 
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上 面 这 段 代 码 在 MyCode\ MySample765\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 外 ,操作 照相 机 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission androlid:name 一 "android. permission.CAMERA"/ > 权限 。 此 实例 的 完整 项 目 在 MyCode 


\MySample765 文件 夹 中 。 
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269 ”从 相册 中 选择 图 像 并 设置 为 手机 壁纸 


此 实例 主要 通过 使 用 setWallpaper() 方 法 ,实现 在 相册 中 选择 图 像 , 并 设置 为 手机 壁纸 。 当 实例 
运行 之 后 , 单 击 “ 从 相册 中 选择 图 像 ” 按 钮 , 则 将 打开 相册 目录 ,然后 在 其 中 任意 选择 一 幅 图 像 , 则 该 图 
像 将 显示 在 ImageView 控件 中 ; 单 击 “ 设 置 为 果 面 壁纸 ”按钮 , 则 显示 的 图 像 将 成 为 介面 壁纸 ,效果 分 
别 如 图 269. 1 的 左 图 和 右 图 所 示 。 
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MySample 


从 相册 中 选择 图 像 ”设置 为 桌面 壁纸 


主要 代码 如 下 : 


void onClickBtnl1(View view){ // 响 应 单 击 "从 相册 中 选择 图 像 "按钮 
myImageView. setDrawingCacheEnabled( true); 
Intent myIntent = new Intent(Intent. ACTION PICK); 
myIntent. setType(" image/ * " ); 
startActivityForResult(myIntent, 1); 
} 
(QOverride 


protected void onActivityResult( int requestCode, 
int resultCode, Intent data){ // 获 取 在 相册 中 选择 的 图 像 


Uri myUri = data. getData( ) ; // 获 取 图 像 数 据 
myImageView. setImageURI(myUri); // 预 览 所 选 图 像 
} 
void onClickBtn2(View view){ // 响 应 单 击 "设置 为 桌面 壁纸 "按钮 
try{ 
Bitmap myBitmap = Bitmap. createBitmap( 
myImageView. getDrawingCache( ) ) ; // 获 取 缓 存 的 图 像 数据 
myImageView. SetDrawingCacheEnabled(false) ; // 清 空 图 像 缓 存 数 据 
setWallpaper (myBitmap); // 设 置 图 像 为 手机 壁纸 


Toast. makeText(MainActivity. this, "设置 壁纸 成 功 !",Toast. LENGTH_SHORT). show(); 
}catch(Exception e){ e.printStackTrace( ); } 
I 
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上 F 面 这 段 代 码 在 MyCode\ MySample657\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 外 ,设置 壁纸 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission android:name 一 "android. permission. SET_WALLPAPER"/ > 权限。 此 实例 的 完整 项 目 
在 MyCode\MySample657 文件 夹 中 。 


270 ”使 用 Runnable 间隔 执行 重复 的 任务 


此 实例 主要 通过 使 用 Runnable, 实 现 间隔 执行 重复 的 任务 。 当 实例 运行 之 后 ,两 幅 电 影 海 报 图 像 
每 间隔 5 秒 相 互 切换 一 次 ,并 且 永 不 停 软 ,效果 分 别 如 图 270. 1 的 左 图 和 右 图 所 示 。 


MySample 


ATI NA 


图 270.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
public TransitionDrawable myTransition; 
private ImageView myImageView; 
boolean bChecked = true; 
final Handler myHandler = new Handler(); 
Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
myHandler. postDelayed( myRunnable, 5000); // 每 间隔 5 秒 执行 一 次 任务 
} 
// 设 置 要 执行 的 任务 
Runnable myRunnable = new Runnable() { 
(QOverride 
public void run() { 
try { 
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myHandler. postDelayed(this, 5000); 


if (bChecked) { // 从 myimagel 切换 到 myimage2 
myTransition = new TransitionDrawable(new Drawable[ ]{ 
getDrawable(R. mipmap. myimage1l ), 
getDrawable(R. mipmap. myimage2 ) } ); 
} else { // 从 myimage2 切换 到 myimagel 


myTransition = new TransitionDrawable(new Drawable[ ]{ 
getDrawable(R. mipmap. myimage2 ), 
getDrawable(R. mipmap. myimagel ) } ); 
} 
myImageView. setImageDrawable(myTransition); 
myTransition. startTransition(5000); 
bChecked = !bChecked; 
} catch (Exception e) { e.printStackTrace( );} 
Pe 


上 面 这 段 代码 在 MyCode\ MySample655\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample655 文件 夹 中 。 


271 使 用 Timer 实现 促销 活动 的 倒计时 功能 


此 实例 主要 通过 使 用 Timer 实现 促销 活动 的 倒计时 功能 。 当 实例 运行 之 后 ,立即 开始 进行 倒 计 
时 ,效果 分 别 如 图 271. 1 的 左 图 和 右 图 所 示 。 


外 口 
) 


MySample 


图 271. 1 
主要 代码 如 下 : 
public class MainActivity extends Activity { 


long myCount = 600; // 总 时 间 是 600 秒 
(@ Override 
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protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
setContentView(R. layout. activity main); 
final TextView myHour = (TextView) findViewById(R. id. myHour); 
final TextView myMinute = (TextView) findViewById(R. id.myMinute); 
final TextView mySecond = (TextView) findViewById(R. id.mySecond); 
final Timer myTimer = new Timer(); 
myTimer. schedule(new TimerTask() { 
@ Override 
public void run() { 
runOnUiThread(new Runnable() { 
(QOverride 
public void run() { 
myCount -一 ; 
String myString = formatLongToTimeStr(myCount); 
String[ ] mySpilt = myString. split(":"); 
for (int i = 0; i< mySpilt. length; i++) { 
myHour. setText (mySpilt[0]); 
myMinute. setText(mySpilt[1]); 


mySecond. setText (mySpilt[2]); 
} 
if (myCount <= 0) { myTimer. cancel(); } 
} a } },1000, 1000); // 每 秒 刷 新 一 次 ， 并 递减 1 秒 
} 
public String formatLongToTimeStr(Long 1) { // 格 式 化 时 间 


int hour = 0; 

int minute = 0; 

int second = 1. intValue( ); 
if (second > 60) { 


minute = second / 60; 


second = Second % 60; 


} 

if (minute > 60) { 

hour = minute / 60; 
minute = minute % 60; 


} 


String myTime = hour + ":" + minute + ":" + second; 
return myTime; 


Le 


上 面 这 上段 代码 在 MyCode\ MySample707\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 实例 的 完整 项 目 在 MyCode\MySample707 文件 夹 中 ， 


272 ”使 用 Runtime 执行 系统 命令 静默 安装 应 用 包 


此 实例 主要 通过 使 用 Runtime. getRuntime(). exec() 方 法 执行 系统 指令 “pm install 一 r”, 实 现 静 
默 安装 指定 的 应 用 安装 包 。 当 实例 运行 之 后 , 单 击 “选择 安装 文件 ”按钮 ,然后 在 弹出 的 "内 部 存储 空 
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间 ” 窗 口中 选择 安装 文件 包 , 即 可 实现 静默 安装 ,效果 分 别 如 图 272. 1 的 左 图 和 右 图 所 示 。 


0 MySample 


正在 申请 Root 授权 


用 途 : 未 知 Root 权限 的 用 途 ， 请 谨慎 
授权 。 


提示 : 若非 本 人 行为 ， 建 议 拒绝 Y 永久 记 住 选择 


口 24 小 时 内 记 住 选择 
拒绝 (6) 


拒绝 (25) 


gm 4 


正在 静默 安装 . 


图 272.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
Handler myHandler = new Handler( ){ 
人 Override 
public void handleMessage( Message msg){ 
if(msg.what == 0){ 
Toast. makeText (MainActivity. this, 
"正在 静默 安装 . . .",Toast. LENGTH SHORT). show( ) ; 
}else if(msg. what == 1){ 
Toast. makeText (MainActivity. this, "安装 失败 !",Toast. LENGTH_SHORT). show( ) ; 
finish( ); 
}else if(msg.what == 2){ 
Toast. makeText (MainActivity. this, "安装 成 功 !",Toast. LENGTH_SHORT). show( ); 
finish( ); 
rks 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 


setContentView(R. layout. activity main) ; 


} 
public void onClickBtnl (View v) { // 响 应 单 击 " 选 择 安 装 文件 "按钮 


Intent myIntent = new Intent(Intent. ACTION_GET_CONTENT ) ; 

// 筛 选 文件 后 缀 名 是 .apk 的 安装 文件 

myIntent. setType( "application/vnd. android. package - archive" ) ; 
myIntent. addCategory( Intent. CRTEGORY_OPENRBLE ) ; 
startActivityForResult(myIntent, 1); // 跳 转 至 文件 选择 界面 
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} 


(Override 
protected void onActivityResult( int requestCode, int resultCode, Intent data){ 
if(resultCode == RESULT_OK) { // 返 回 用 户 所 选 APK 文件 信息 
myHandler. sendEmptyMessage( 0); // 开 始 静 默 安装 操作 


Uri myUri = data. getData( ) ; 
final String myPath = getRealPathFromURI(myUri) ; // 将 Uri 转换 为 绝对 路 径 
myHandler. postDelayed(new Runnable( ){ 


(@ Override 
public void run( ){ // 进 行 静默 安装 并 返回 安装 结果 
boolean myInstallStatus = silenceInstall (myPath); 
if(!'myInstallStatus){ // 表 示 安 装 失败 
myHandler. sendEmptyMessage(1); 
}else{ // 表 示 安 装 成 功 
myHandler. sendEmptyMessage(2); 
} } },1000); // 使 用 延 时 操作 是 为 了 等 待 应 用 返回 主 界面 ,否则 会 黑屏 


本 
public String getRealPathFromURI(Uri uri){ 

String myPath = null; 

String[ ] myProjection = {MediaStore. Images. Media. DATA}; 

Cursor myCursor = getContentResolver().query(uri,myProjection, null,null, null); 

// 在 数据 库 中 查询 指定 Uri, 并 返回 符合 条 件 的 文件 路 径 

if(myCursor. moveToFirst()){ 
int myIndex = myCursor. getColumnIndexOrThrow(MediaStore. Images. Media. DATA); 
myPath = myCursor. getString(myIndex); 

} 

myCursor. close( ); 

return myPath; 

} 
public boolean silenceInstal1(String apkPath){ 

boolean myResult = false; 

Data0utputStream myOutputStream = null; 

BufferedReader myErrorReader = null; 

try{ 
Process myProcess = Runtime. getRuntime( ).exec(" su" ); 
myOutputStream = new Data0utputStream(myProcess. getOutputStream( ) ) ; 
String myCommand = "pm install - 工 " + apkPath + "\n"; 
myOutputStream. write(myCommand. getBytes(Charset. forName("utf - 8" ))); 
myOutputStream. flush( ); 
myOutputStream. writeBytes("exit\n" ); 
myOutputStream. flush( ); 
myProcess. waitFor( ); 
myErrorReader = new BufferedReader( 

new InputStreamReader (myProcess. getErrorStream( ) ) ) ; 
String myInstallMessage = "",myMessageLine; 
// 读 取 命 令 的 执行 结果 
while( (myMessageLine = myErrorReader. readLine())!= null){ 
myInstallMessage += myMessageLine; 

} 
if (!myInstallMessage. contains("Failure" )) myResult = true; 

}catch(Exception e){ } 

finally{ 
try{ 
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myOutputStream. close(); 
myErrorReader. close(); 


} catch(Exception e){ } 
} 
return myResult; 
}]} 


上 面 这 段 代 码 在 MyCode\ MySample873\app\src\ main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 此 外 , 读 取 SD 卡 文件 需要 在 AndroidManifest. xml 文件 中 添加 < uses- 
permission androlid:name 一 "android. permission. READ EXTERNAL _ STORAGE"/ > 权限 。 此 实例 
的 完整 项 目 在 MyCode\MySample873 文件 夹 中 ， 
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273 ”使 用 腾讯 SDK 获取 授权 QQ 账户 的 简介 


此 实例 主要 通过 使 用 腾讯 SDK ,实现 在 登录 QQ 账户 成 功 之 后 ,从 当前 应 用 获取 QQ 个 人 账户 的 
简介 信息 。 当 实例 运行 之 后 , 单 击 “ 登 录 QQ 账户 ”按钮 , 则 自动 跳 转 到 QQ 登录 界面 执行 登录 QQ 账 
户 的 操作 ,如 果 登 录 成 功 , 则 在 弹出 的 Toast 中 显示 “登录 成 功 !”, 如 图 273. 1 的 左 图 所 示 。 只 有 在 合 
录 QQ 账户 成 功 之 后 , 单 击 “ 获 取 账 户 信息 ”按钮 ,才能 在 弹出 的 Toast 中 显示 该 QQ 账户 的 简介 信 
息 ,如 图 273. 1 的 右 图 所 示 。 


MySample MySample 


获取 账户 信息 登录 QQ 账户 获取 账户 信息 


用 户 了 昵称 : 罗 帅 

用 尸 性 别 : 男 

用 户 所 在 地 : 重庆 渝 北 
黄金 会 员 等 级 : 0 

会 员 等 级 : 0 


图 273. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements IUiListener { 
Tencent myTencent; 
人 Override 
protected void onCreate(Bundle savedInstanceState) { 
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super. onCreate( savedInstanceState) ; 
myTencent = Tencent.createlInstance("1106416651", getApplicationContext( )); 
setContentView(R. layout. activity main); 


} 


public void onClickBtn1 (View v) { // 响 应 单 击 " 登 录 QQ 账户 "按钮 
myTencent. login(MainActivity. this, "all", this); // 使 用 QQ 账号 进行 授权 登录 
} 
public void onClickBtn2 (View v) { // 响 应 单 击 "获取 账户 信息 "按钮 
UserInfo myUserInfo = new UserInfo(this，myTencent. getQQToken( ) ) ; 
myUserInfo. getUserInfo(new IUiListener() { // 通 过 回调 函数 解析 用 户 信 息 
(Override 
public void onComplete(Object o) { 
try { 


JSONObject myJsonObject = new JSONObject(o.toString( )); 
String myNickName = myJsonObject.getString("nickname" ); 
String myGender = myJsonObject.getString(" gender" ); 
String myProvince = myJsonObject.getString("province" ); 
String myCity = myJsonObject.getString("city" ); 
String myYellowVipLevel = myJsonObject.getString("yellow vip level" ); 
String myVipLevel = myJsonObject.getString(" level" ); 
// 使 用 StringBuilder 拼接 字符 串 内 容 
StringBuilder myBuilder = new StringBuilder(); 
myBuilder.append(" 用 户 上 昵称 :" + myNickName + "\n"); 
myBuilder.append(" 用 户 性 别 :" + myGender + "\n"); 
myBuilder. append(" 用户 所 在 地 :" + myProvince + " " + myCity + "\n"); 
myBuilder.append(" 黄 金 会 员 等 级 :" + myYellowVipLevel + "\n"); 
myBuilder. append(" 会 员 等 级 :" + myVipLevel + "\n"); 
Toast. makeText(MainActivity. this, 
myBuilder. toString( ), Toast.LENGTH SHORT). show(); 
} catch (Exception e) { e.PprintStackTrace( ); } 
} 
(Override 
public void onError(UiError uiError) { } 
(Override 
public void onCancel() { } 
}); } 
(Override 
public void onComplete(Object o) { 
try { 
// 在 进行 授权 登录 操作 时 ,获取 其 access_token、openid 和 expire_in 值 ， 
// 并 通过 这 些 字段 值 来 获取 授权 用 户 信 息 
JSONObject myJsonObject = new JSONObject(o.toString( ) ) ; 
myTencent. setAccessToken(myJsonObject. getString(" access token" ), 
myJsonObject. getString(" expires in" )); 
myTencent. setOpenId(myJsonObject. getString(" openid" ) ); 
Toast. makeText (MainActivity.this, "登录 成 功 !"，, Toast.LENGTH SHORT). show( ); 
} catch (Exception e) { e.printStackTrace( ); } 
} 
(Override 
public void onError(UiError uiError) { } 
(Override 
public void onCancel() { } 
(Override 
protected void onActivityResult( int requestCode, 
int resultCode，Intent data) { // 处 理 登 录 回 调 操 作 
Tencent. onActivityResultDatal( requestCode, resultCode, data, this); 


} } 
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上 F 面 这 段 代 码 在 MyCode\ MySample939\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 .此 实例 需要 在 MyCode\MySample939\app\libs 目录 下 添 
加 库 文 件 open_sdk r5886 lite. jar, 该 库 文件 的 下 载 地 址 是 http://wiki. connect. qq. com/sdk%E4% 
B8%8B%E8%BD%BD。 然 后 在 MyCode\MySample939\app 目录 下 的 build. gradle 文件 的 根 节点 下 
添加 如 下 代码 : 


repositories { flatDir { dirs 'libs' } } 


并 在 该 文件 的 dependencies 节点 下 添加 依赖 项 : 


compile files('libs/open sdk r5886 lite. jar') 
compile 'com. android. support: support ~ v4:26.0.0— alphal' 


同时 还 要 在 MyCode\MySample939\app\src\main\AndroidManifest. xml 文件 中 按照 下 列 粗 体 
字 所 示 的 内 容 进 行 修改 : 


<?xml version = "1.0”encoding = "utf 一 8"?> 
< manifest xmlns:android = "http://schemas. android. com/apk/res/android" 
package = "com. bin. luo. mysample"> 
<application 
android:allowBackup = "true" 
android: icon = "(@mipmap/ic_launcher" 
android: label = "(@ string/app_name" 
android: supportsRtl] = "true" 
android:theme = "(@ style/AppTheme"> 
< activity android:name = ". MainActivity"> 
< intent ~ filter> 
< action android:name = "android. intent. action.MRIN" /> 
< category android:name = "android. intent. category. LAUNCHER" /> 
</intent -filter > 
</activity> 
<activity 
android: name = "com. tencent. tauth. AuthActivity" 
android: launchMode = " singleTask" 
android:noHistory = "true"> 
< intent - filter> 
< action android:name = "android. intent. action. VIEW" /> 
<category android: name = "android. intent. category. DEFAULT" /> 
<category android:name = "android. intent. category. BROWSABLE" /> 
< data android: scheme = "tencent1106416651"/><! -- 填 入 你 的 APP ID--> 
</ intent - filter > 
</activity> 
<activity 
android: name = "com. tencent. connect. common. AssistActivity" 
android: configChanges = "orientation|keyboardHidden | screenSize" /> 
</application> 
<uses - permission android: name = "android. permission. INTERNET" /> 
< uses - permission android: name = "android. permission. ACCESS NETWORK STATE" /> 
</manifest > 


此 外 ,使 用 腾讯 SDK 服务 需要 在 腾讯 开放 平台 上 创建 应 用 并 获取 AppID, 步 又 如 下 。 
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(1) 登录 腾讯 开放 平台 (http://open. qq. com) 并 进入 管理 中 心 页 面 。 

(2) 单 击 右 侧 的 “创建 应 用 ”按钮 ,然后 选择 “移动 应 用 安 单 ”选项 ,再 单 击 “创建 应 用 ”按钮 进行 下 
一 步 操 作 。 在 弹出 的 对 话 框 中 选择 “软件 ?选项 ,并 单 击 “确定 ”按钮 进行 下 一 步 操 作 。 

(3) 在 表单 中 填写 应 用 的 相关 信息 ,填写 完成 后 单 击 底部 的 “提交 审核 ”按钮 即 可 完成 应 用 创建 操 
作 。 应 用 审核 一 般 会 在 24 小 时 内 完成 并 反馈 ID 号 码 , 即 AndroidManifest. xml 文件 中 的 ID 号 。 

此 实例 的 完整 项 目 在 MyCode\MySample939 文件 夹 中 。 


274 ”使 用 腾讯 SDK 实现 以 第 三 方 登录 QQ 账户 


此 实例 主要 通过 使 用 腾讯 SDK ,实现 以 第 三 方形 式 登 录 QQ 账户 。 当 实例 运行 之 后 , 单 击 “ 登 录 
QQ 账户 ”按钮 , 则 将 显示 QQ 登录 窗口 ,如 果 登 录 成 功 ,将 在 弹出 的 Toast 中 显示 登录 结果 ,效果 分 别 
如 图 274. 1 的 左 图 和 右 图 所 示 。 
G520 mm +S, 


QQ 登录 MySample 


Android 精 彩 实 例 集锦 


加 他 
2099904576 


添加 帐号 


274.1 
主要 代码 如 下 : 


public class MainActivity extends Activity implements IUiListener { 

Tencent myTencent; 

Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
//1106416651 是 腾讯 反馈 给 申请 人 的 ID, 各 个 应 用 都 不 相同 
myTencent = Tencent.createInstance("1106416651", getApplicationContext( )); 
setContentView(R. layout. activity main); 

} 

public void onClickBtnl (View v) { // 响 应 单 击 "登录 QQ 账户 "按钮 

myTencent. login(MainActivity.this, "all", this); 

} 

@Override 
protected void onActivityResult( int requestCode, 
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int resultCode, Intent data) { // 处 理 登 录 回 调 操 作 
Tencent. onActivityResultData( requestCode, resultCode, data, this); 
} 


(Override 

public void onComplete(Object o) { // 登 录 成 功 时 的 回调 操作 
Toast. makeText (MainActivity.this, "登录 成 功 !"，Toast.LENGTH_SHORT) . show( ); 

} 

(Override 

public void onError(UiError uiError) { // 登 录 失 败 时 的 回调 操作 
Toast. makeText (MainActivity. this, "登录 失败 !" ，Toast. LENGTH SHORT ) . show( ) ; 

} 

(Override 

public void onCancel() { // 登 录取 消 时 的 回调 操作 


Toast. makeText (MainActivity.this, 
"取消 登录 操作 !"， Toast.LENGTH SHORT). show( ); 


} } 


上 面 这 段 代 码 在 MyCode\ MySample942\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 参考 实例 273 (或 直接 看 源 代码 ) 的 内 容 在 
MyCode\ MySample942\app\libs 目录 下 添加 库 文 件 open_sdk_ r5886 lite. jar, 及 修改 MyCode\ 
MySample942\app\build. gradle 文件 和 MyCode\MySample942\app\src\main\ AndroidManifest. 
xml 文件 。 此 实例 的 完整 项 目 在 MyCode\MySample942 文件 夹 中 。 


275 ”使 用 腾讯 SDK 将 指定 文本 分 至 给 QQ 好 友 


此 实例 主要 通过 使 用 腾讯 SDK ,实现 将 指定 的 文本 内 容 分 享 给 QQ 好 友 。 当 实例 运行 之 后 ,在 各 
个 输入 框 中 输入 对 应 的 内 容 , 如 图 275. 1 的 左 图 所 示 ; 然后 单 击 “ 分 享 给 QQ 好 友 ” 按 钮 , 则 将 启用 
QQ 分 享 功能 ,如 图 275. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickBtnl (View v) { // 啊 应 单 击 "分 享 给 QQ 好 友 " 按 钮 
Bundle myBundle = new Bundle( ); 
myBundle. putInt (QQShare. SHARE TO QQ KEY TYPE, 


QQShare. SHARE TO QQ TYPE APP); // 设 置 分 享 类 型 为 应 用 分 享 类 型 
myBundle. putString(QQShare. SHARE TO QQ TITLE, 

myEditTitle. getText().toString() ); // 设 置 分 享 标题 
myBundle. putString(QQShare. SHARE TO QQ SUMMARY, 

myEditSummary. getText().toString( )); // 设 置 分 享 摘要 
myBundle. putString(QQShare. SHARE TO QQ APP NAME, 

myEditAppname. getText().toString( )); // 设 置 分 享 应 用 的 显示 名 称 
myTencent. shareToQQ(MainActivity.this, myBundle, this);  // 分 享 应 用 至 QQ 好 友 


} 


上 面 这 上段 代码 在 MyCode\ MySample940\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 参考 实例 273( 或 直接 看 源 代码 ) 的 内 容 在 
MyCode\ MySample940\app\libs 目录 下 添加 库 文 件 open_sdk_r5886_lite. jar, 及 修改 MyCode\\ 
MySample940\app\build. gradle 文件 和 MyCode\MySample940\app\src\main\ AndroidManifest. 
xml 文件 。 此 实例 的 完整 项 目 在 MyCode\MySample940 文件 夹 中 。 


276 ”使 用 腾讯 SDK 将 本 地 图 像 发 表 到 QQ 至 间 


此 实例 主要 通过 使 用 腾讯 SDK ,实现 将 本 地 图 像 发 表 到 QQ 空间 。 当 实例 运行 之 后 , 单 击 * 选 择 
图 像 文 件 ? 按 钮 ,然后 在 弹出 的 窗口 中 选择 图 像 文件 , 则 对 应 的 图 像 将 显示 在 ImageView 控件 中 ,如 
图 276. 1 的 左 图 所 示 ; 然后 单 击 " 发 表 到 QQ 空间 ”按钮 , 则 将 跳 转 到 发 送 窗口 ; 在 发 送 窗口 中 单 击 
“发 送 ” 按 钮 , 则 该 图 像 将 会 出 现在 QQ 空间 中 ,如 图 276. 1 的 右 图 所 示 。 
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主要 代码 如 下 : 


public class MainActivity extends Activity implements IUiListener { 
Tencent myTencent.; 
String myPath = Environment. getExternalStorageDirectory() + "/myimg2. jpg"; 
ImageView myImageView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
//1106416651 是 腾讯 反馈 给 申请 人 的 ID, 各 个 应 用 都 不 相同 
myTencent = Tencent. createInstance("1106416651", getApplicationContext( )); 
setContentView(R. layout.activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 
} 
public void onClickBtnl (View v) { // 响 应 单 击 " 选 择 图 像 文件 "按钮 
Intent myIntent = new Intent(Intent.ACTION PICK); 
myIntent. setType("image/ * "); 
// 打 开 相 册 窗 口 ,并 返回 所 选 图 像 文件 路 径 
startActivityForResult(myIntent, 1); 
} 
public void onClickBtn2 (View v) { // 响 应 单 击 "发 表 到 QQ 空间 "按钮 
final Bundle myBundle = new Bundle( ); 
myBundle. putInt(QQShare. SHARE TO QQ KEY TYPE， 
QQShare. SHARE TO QQ TYPE IMAGE); 
myBundle. putString( QQShare. SHARE TO QQ IMAGE LOCAL URL, myPath); 
myBundle. putString( QQShare. SHARE TO Q0 APP NAME, "测试 应 用 " ); 
myBundle. putInt (QQShare. SHARE TO QQ ExXT INT, 
QQOShare. SHARE TO QQ FLAG QZONE AUTO OPEN); 
myTencent. shareToQQ( MainActivity.this, myBundle, this); 
} 
public static String getRealFilePath(final Context context, 
final Uri uri) { // 获 取 图 像 文件 的 路 径 
if (null == uri) return null; 
final String scheme = uri.getScheme( ); 
String data = null; 
if (scheme == null) data = uri. getPath( ) ; 
else if (ContentResolver. SCHEME FILE. equals(scheme)){data = uri.getPath();} 
else if (ContentResolver. SCHEME CONTENT. equals(scheme)) { 
Cursor cursor = context. getContentResolver().query(uri, 
new String[ ] {MediaStore. Images. ImageColumns. DATA}, null, null, null]l); 
if (null != cursor) { 
if (cursor.moveToFirst()) { 
int index = cursor.getColumnIndex(MediaStore. Images. ImageColumns. DATA); 
if (index > -1) { data = cursor.getString( index); } 
} 


cursor. Close( ); 


上 
return data; 
} 
(Override 
protected void onActivityResult( int requestCode, 
int resultCode, Intent data) { 


if (resultCode == RESULT OK) { 
Uri myUri = data. getDatal( ); 
myImageView. setImageURI(myUri); // 显 示 所 选 图 像 文件 


myPath = getRealFilePath(MainActivity. this, myUri); 
} else {Tencent. onActivityResultDatal( requestCode, resultCode, data, this);} 
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} 
(Override 
public void onComplete(Object o) { // 发 表 成 功 时 的 回调 函数 
Toast. makeText (MainActivity.this, "发 表 成 功 !", Toast. LENGTH SHORT) . show( ) ; 
} 
(Override 
public void onError(UiError uiError) { // 发 表 失 败 时 的 回调 函数 


Toast. makeText (MainActivity. this, "发 表 失 败 !" 
+ uiError.errorMessage, Toast. LENGTH SHORT). show( ); 


} 
(QOverride 
public void onCancel() { // 取 消 发 表 时 的 回调 函数 
Toast. makeText (MainActivity. this, "取消 发 表 !",Toast. LENGTH_ SHORT). show( ); 
1 


上 面 这 段 代 码 在 MyCode\ MySample948\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 参考 实例 273( 或 直接 看 源 代 码 ) 的 内 容 在 
MyCode\ MySample948\app\libs 目录 下 添加 库 文 件 open_sdk r5886 lite. jar, 及 修改 MyCode\ 
MySample948\app\build. gradle 文件 和 MyCode\MySample948\app\src\main\ AndroidManifest. 
xml 文件 。 此 实例 的 完整 项 目 在 MyCode\MySample948 文件 夹 中 。 


277 ”使 用 微 信 SDK 将 视频 链接 分 人 享 给 微 信 好 友 


此 实例 主要 通过 使 用 微 信 SDK 的 WXMediaMessage 和 SendMessageToWX, 实 现 将 指定 的 视频 
链接 分 享 到 指定 的 微 信 好 友 对 话 中 。 当 实例 运行 之 后 ,在 “视频 文件 链接 : ”输入 框 中 输入 视频 链接 ， 
如 “http://v. youku. com/v_show/id XMzA5MDk5OTcxMg= =. html? spm= a2hww. 20027244. 
m_250166. 5 一 5! 2 一 5 一 5 一 5 一 5 一 A”, 如 图 277. 1 的 左 图 所 示 ; 然后 单 击 “将 下 面 的 视频 链接 分 享 
至 微 信 和 好友” 按钮 , 则 该 视频 链接 内 容 将 出 现在 指定 的 微 信 好 友 对 话 中 ,如 图 277. 1 的 右 图 所 示 。 
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图 277. 1 
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主要 代码 如 下 : 


public class MainActivity extends Activity { 
IWXAPI myWeChat; 
EditText myEditText; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
myWeChat = WXAPIFactory.createWXAPI(this, 


"wx9fbf8c966226923a", true); // 创 建 微 信和 接口 对 象 
myWeChat. registerApp("wx9fbf8c966226923a" ) ; // 将 应 用 的 AppID 注册 至 微 信 


setContentView(R. layout. activity main); 
myEditText = (EditText) findViewById(R. id. myEditText) ; 

} 

// 响 应 单 击 " 将 下 面 的 视频 链接 分 享 至 微 信 好 友 " 按 钮 

public void onClickBtnl (View v) { 
WXVideoObject myVideoObject = new WXVideoObject( ); 
// 设 置 分 享 视频 url 链接 
myVideoObject. videoUr1l = myEditText.getText().toString(); 
WZXMediaMessage myWXMediaMessage = new WXMediaMessage(myVideoObject); 
myWXMediaMessage.title =" 震 撼 大 片 :领航 "; // 设 置 分 享 视频 标题 
SendMessageToWX. Req myRequest = new SendMessageToNX. Req(); 
// 设 置 请 求 唯一 标识 符 
myRequest. transaction = String.valueOf(System.currentTimeMillis( )); 
myRequest. message = myWXMediaMessage; 
myRequest. scene = SendMessageToWX. Req.WXSceneSession; // 微 信 好 友 对 话 
myWeChat. sendReq( myRequest ); // 发 起 分 享 请 求 

}} 


面 这 段 代 码 在 MyCode\ MySampleX88\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 在 MyCode\MySampleX88\app\ build. gradle 
文件 中 添加 compile 'com. tencent. mm. opensdk :wechat 一 sdk 一 android 一 without 一 mta: 十 ' 依 赖 项 ， 
同时 还 要 在 MyCode\MySampleX88\app\src\main\AndroidManifest. xml 文件 中 添加 下 列 权 限 : 


< uses - permission android:name = "android. permission. INTERNET" /> 

< uses - permission android:name = "android. permission. ACCESS NETWORK STATE"/> 
< uses - permission android:name = "android. permission. ACCESS WIFI STATE"/> 

< uses - permission android:name = "android. permission. READ PHONE STRTE" /> 


此 外 ,使 用 微 信 SDK 需要 在 微 信 开放 平台 上 创建 应 用 并 获取 AppID, 具 体 步 骤 如 下 。 

(1) 登录 微 信 开放 平台 账号 (https://open. weixin. qq. com) ,进入 “管理 中 心 ”, 单 击 “ 创 建 移动 应 
用 ”按钮 创建 新 应 用 。 

(2) 在 表单 内 填写 应 用 的 相关 信息 ,完成 后 单 击 “ 下 一 步 ” 按 钮 。 

(3) 在 应 用 平台 选项 中 选择 “Android 应 用 ”, 并 在 其 下 方 表单 内 填 入 相关 信息 ,完成 后 单 击 “提交 
审核 ”按钮 即 可 完成 应 用 创建 操作 。 一 般 应 用 审核 需要 7 个 工作 日 左右 时 间 , 其 中 应 用 签名 需要 通过 
签名 生成 工具 来 获取 ,该 工具 可 在 微 信 开放 平台 的 资源 中 心 下 载 。 

(4) 应 用 创建 审核 通过 后 ,会 自动 生成 AppID 值 。 用 户 需 要 将 该 AppID 值 传 人 微 信 接口 对 象 构 
造 器 中 以 调用 微 信 SDK 相关 功能 。 详 细 内 容 请 参考 官方 网 站 (https:// open. weixin. qq. com ) 。 

此 实例 的 完整 项 目 在 MyCode\MySampleX88 文件 夹 中 ， 
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278 ”使 用 微 信 SDK 将 音乐 链接 分 于 到 朋友 图 


此 实例 主要 通过 使 用 微 信 SDK 的 WXMediaMessage 和 SendMessageToWX, 实 现 将 指定 的 音乐 
链接 分 享 到 微 信 的 朋友 圈 。 当 实例 运行 之 后 ,在 “音乐 文件 链接 : “输入 框 中 输入 音乐 链接 ,如 
“http://staff2. ustc. edu. cn /一 wdw /softdown/index. asp/0042515 05. ANDY. mp3”, 如 图 278. 1 
的 左 图 所 示 ; 然后 单 击 “ 将 下 面 的 音乐 链接 分 享 至 微 信 朋友 圈 ? 按 钮 , 则 该 音乐 链接 将 出 现在 微 信 的 朋 
友 圈 中 ,如 图 278. 1 的 右 图 所 示 。 
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// 啊 应 单 击 "将 下 面 的 音乐 链接 分 享 至 微 信 朋 友 圈 "按钮 
public void onClickBtnl (View v) { 
WXMusicObject myMusicObject = new WXMusicObject(); 
// 设 置 分 享 音乐 的 url 链接 
myMusicObject. musicUrl = myEditText.getText().toString(); 
WZXMediaMessage myMessage = new WxXMediaMessage(myMusicObject); 
myMessage. mediaObject = myMusicObject; 
myMessage.title = "音乐 分 享 测试 "; // 设 置 音乐 链接 分 享 标题 
// 设 置 音乐 链接 分 享 缩 略图 样式 
myMessage. setThumbImage( BitmapFactory. decodeResource( getResources( ) ， 
R. mipmap. ic launcher)); 
SendMessageToWX. Req myRequest = new SendMessageToNX. Req(); 
// 设 置 该 请 求 对 象 所 对 应 的 唯一 标识 符 
myRequest. transaction = String.valueOf(System. currentTimeMillis( )); 
myRequest. message = myMessage; 
myRequest. scene = SendMessageToWX. Req. WXSceneTimeline; // 微 信 朋 友 圈 
myWeChat. sendReq( myRequest ); // 发 送 分 享 请 求 
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上 面 这 上段 代码 在 MyCode\ MySample926\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 在 MyCode\MySample926\app\ build. gradle 
文件 中 添加 compile 'com. tencent. mm. opensdk :wechat 一 sdk 一 android 一 without 一 mta: 十 ' 依 赖 项 ， 
同时 还 要 在 MyCode\MySample926\app\src\main\ AndroidManifest. xml 文件 中 添加 下 列 权限 : 


<uses— permission android:name = "android. permission. INTERNET" /> 

<uses— permission android:name = "android. permission. ACCESS NETWORK_ STATE"/> 
<uses— permission android:name = "android. permission. ACCESS WIFI_STATE"/> 
<uses— permission android:name = "android. permission. READ PHONE STATE"/> 


此 外 ,使 用 微 信 SDK 需要 在 微 信 开放 平台 上 创建 应 用 并 获取 AppID, 具 体 步 又 请 参考 官方 网 站 
(https://open. weixin. gq. com) 。 此 实例 的 完整 项 目 在 MyCode\MySample926 文件 夹 中 。 


279 ”使 用 百度 SDK 根据 起 点 和 终点 规划 步行 线路 


此 实例 主要 通过 使 用 百度 地 图 SDK ,实现 根据 起 点 和 终点 规划 步行 线路 。 当 实例 运行 之 后 ,在 
“城市 : “输入 框 中 输入 城市 名 称 , 如 “北京 ”, 在 “起 点 : “输入 框 中 输入 起 点 名 称 , 如 “清华 大 学 ”, 在 “ 终 
点 : "输入 框 中 输入 终点 名 称 , 如 "北京 动物 园 ”, 然 后 单 击 “规划 步行 线路 "按钮 , 则 在 地 图 中 显示 起 点 
和 终点 之 间 的 步行 线路 ; 单 击 该 线路 上 的 中 转 站 图 标 , 则 在 弹出 的 Toast 中 显示 在 该 中 转 站 点 的 步行 
建议 ,效果 分 别 如 图 279. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 


城市 : 北京 规划 步行 线路 城市 : 北京 规划 步行 线路 
清华 大 学 终点 : 北京 动物 园 起 点 : 国家 大 剧院 终点: 公安 部 


六 圆明园 遗址 公园 \ 


入 大 
此 处 建议 : 沿 中 关 村 东 路 辅路 走 760 米 ， 此 处 建议 : 治 前 门 东 大 街 走 270 米 , 直 走 
直 走 过 人 行道 ,继续 向 前 进入 前 门 东 大 街 辅路 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
BaiduMap myBaiduMap; 
MapView myMapView; 
EditText myEditEnd; 
EditText myEditBegin; 
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EditText myEditCity; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
SDKInitializer. initialize(getApplicationContext( )); 
setContentView(R. layout. activity_main); 
myMapView = (MapView) findViewById(R. id.myMapView); 
myBaiduMap = myMapView.getMap(); 
myEditBegin = (EditText) findViewById(R. id. myEditBegin); 
myEditEnd = (EditText) findViewById(R. id.myEditEnd); 
myEditCity = (EditText) findViewById(R. id.myEditCity); 
} 
public void onClickBtnl (View v) { // 响 应 单 击 " 规 划 步 行 线路 "按钮 
RoutePlanSearch myRoutePlanSearch = RoutePlanSearch. newInstance( ); 
myRoutePlanSearch. setOnGetRoutePlanResultListener( 
new OnGetRoutePlanResultListener() { 
(Override 
public void onGetWalkingRouteResult(WalkingRouteResult result) { 
if (result.getRouteLines() == null) { 
Toast. makeText (MainActivity. this, 
"抱歉 ,未 搜索 到 相关 路 线 !" ，Toast.LENGTH SHORT). show( ); 


} else { 
myBaiduMap. clear(); // 重 置地 图 ,防止 路 线 重 等 
MyRouteOverlay myRouteOverlay = new MyRouteOverlay(myBaiduMap); 
// 将 路 线 数据 传递 至 图 层 
myRouteOverlay. setData( result. getRouteLines().get(0)); 
myRouteOverlay. addToMap( ); // 将 路 线 图 显示 在 地 图 上 
myRouteOverlay. zoomToSpan( ); // 自 适应 缩放 
// 设 置 路 线 节 点 点 击 监听 
myBaiduMap. setOnMarkerClickListener(myRouteOverlay); 
县， 
(Override 
public void onGetTransitRouteResult(TransitRouteResult transitRouteResult){} 
(Override 


public void onGetMassTransitRouteResult(MassTransitRouteResult 
massTransitRouteResult) { } 
@Override 
public void onGetDrivingRouteResult(DrivingRouteResult result) { } 
(@ Override 
public void onGetIndoorRouteResult( IndoorRouteResult indoorRouteResult){} 
(Override 
public void onGetBikingRouteResult(BikingRouteResult bikingRouteResult){} 
3 
// 根 据 地 址 信息 初始 化 起 点 对 象 
PlanNode myStartPoint = PlanNode.withCityNameAndPlaceName( 
myEditCity.getText().toString(), myEditBegin.getText().toString( )); 
// 根 据 地 址 信息 初始 化 终点 对 象 
PlanNode myEndPoint = PlanNode.withCityNameAndPlaceName( 
myEditCity. getText().toString(), myEditEnd.getText().toString( )); 
// 开 始 搜索 最 佳 步 行路 线 , 并 将 路 线 显示 在 地 图 上 
myRoutePlanSearch. walkingSearch( 
(new WalkingRoutePlan0ption( )).from(myStartPoint).to(myEndPoint)); 
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class MyRouteOverlay extends WalkingRouteOverlay { 
public MyRouteOverlay(BaiduMap baiduMap) { super(baiduMap); } 
@Override 
public boolean onRouteNodeClick(int i) {  // 通 过 Toast 显示 步行 建议 信息 
Toast. makeText(MainActivity.this, "此 处 建议 :”+ mRouteLine. getAllStep(). 
get(i).getInstructions(), Toast.LENGTH SHORT) . show(); 
return true; 
1 


上 面 这 段 代 码 在 MyCode\ MySample922\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySample922\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 需 要 说 明 的 是 ， 
此 实例 需要 引入 百度 地 图 组 件 , 即 MyCode\MySample922\app\libs 目录 下 的 所 有 内 容 , 有 了 这 些 文 
件 之 后 ,就 不 再 需要 安装 百度 地 图 App, 即 使 在 纯净 的 模拟 器 环境 中 也 能 进行 正常 测试 。 然 后 在 
MyCode\MySample922\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_Android. jar') 和 
compile 'com. google. android. gms:play-services-appindexing:8. 4.0' 依 赖 项 。 此 外 ,还 要 按照 下 面 粗 
体 字 所 示 的 内 容 修 改 AndroidManifest. xml 文件 : 


<?xml] version= "1.0" encoding = "utf — 8"?> 
< manifest xmlns:android = "http://schemas.android. com/apk/res/android" 
package = "com. bin. luo. mysample"> 
<application 
android:allowBackup = "true" 
android: icon = "(Omipmap/ic_launcher" 
android: label = "(@ string/app_name" 
android:supportsRt1 = "true" 
android:theme = "(@style/AppTheme"> 
< meta - data 
android:name = "com. baidu. lbsapi. API KEY" 
android: value = "21xq8w6f8yS6kmvzLxM8MeBOc1k0TXIX" /> 
<activity android:name = ". MainActivity"> 
< intent ~ filter> 
<action android:name = "android. intent. action. MAIN"/> 
< category android:name = "android. intent. category. LAUNCHER" /> 
</ intent - filter > 
</activity> 
</application> 
<uses - permission android: name = "android. permission. ACCESS NETWORK STATE" /> 
<uses - permission android: name = "android. permission. INTERNET" /> 
<uses -~ permission 
android: name = "com. android. launcher. permission. READ SETTINGS" /> 
<uses - permission android:name = "android. permission. WAKE LOCK"/> 
< uses - permission android: name = "android. permission.CHANGE WIFI STATE"/> 
<uses - permission android:name = "android. permission. ACCESS WIFI STATE" /> 
< uses - permission android: name = "android. permission.GET TASKS"/> 
<uses - permission android: name = "android. permission. WRITE EXTERNAL STORAGE" /> 
<uses - permission android: name = "android. permission. WRITE SETTINGS"/> 
</manifest > 


百度 地 图 开发 说 明 请 参考 官方 网 址 : http://lbsyun. baidu. com/sdk。 此 实例 的 完整 项 目 在 
MyCode\MySample922 文件 夹 中 ，。 
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280 ”使 用 百度 SDK 实现 将 驾车 线路 分 享 给 好 友 


此 实例 主要 通过 使 用 百度 地 图 SDK 的 ShareUrlSearch ,实现 将 指定 起 点 和 终点 的 驾车 线路 分 享 
给 好 友 。 当 实例 运行 之 后 ,在 "起 点 纬度 :” 输 入 框 中 输入 起 点 纬度 值 , 如 "29. 722571”, 在 "起 点 经 度 :” 
输入 框 中 输入 起 点 经 度 值 , 如 “106. 590326”, 在 "终点 纬度 : ”输入 框 中 输入 终点 纬度 值 , 如 
“29. 919309”, 在 “终点 经 度 : ”输入 框 中 输入 终点 经 度 值 ,如 “107. 248061”, 然 后 单 击 “显示 驾车 线路 ” 
按钮 , 则 在 百度 地 图 中 显示 指定 起 点 和 终点 之 间 的 驾车 线路 ,如 图 280. 1 的 左 图 所 示 。 单 击 分享 驾 
车 线路 ”按钮 , 则 将 把 指定 起 点 和 终点 之 间 的 驾车 线路 (地 址 短 串 ) 分 享 到 第 三 方 应 用 ,如 QQ 好 友 等 ， 


如 图 280. 1 的 右 图 所 示 。 
中 汪 
MySample 
起 点 纬度 : 29.722571 
起 点 经 度 : 106.590326 
终点 纬度 : 29.919309 
终点 经 度 : 107.248061 


显示 驾车 线路 分 享 驾车 线路 发 送 给 渝 北 万 博 


的 家 多 通过 百度 地 图 分 享 的 驾车 线路 是 : http:// 
高 难 镇 j.map.baidu.com/S/ALeRXLV 


取消 发 送 


主要 代码 如 下 : 


public class MainActivity extends Activity { 

BaiduMap myBaiduMap; 

MapView myMapView; 

EditText myEditPoint1A; 

EditText myEditPoint1B; 

EditText myEditPoint2A; 

EditText myEditPoint2B; 

PlanNode myStartNode; 

PlanNode myEndNode ; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
SDKInitializer. initializel(getApplicationContext( ) ) ; 
setContentView(R. layout. activity_ main) ; 
myMapView = (MapView) findViewById(R. id.myMapView); 
myBaiduMap = myMapView. getMap( ) ; 
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myEditPointlA = (EditText) findViewById(R. id.myEditPoint1A); 
myEditPoint1B (EditText) findViewById(R. id.myEditPoint1B); 
myEditPoint2A (EditText) findViewById(R. id.myEditPoint2A); 
myEditPoint2B = (EditText) findViewById(R. id.myEditPoint2B); 
} 
public void onClickBtnl (View v) { // 响 应 单 击 "显示 驾车 线路 "按钮 
RoutePlanSearch myRoutePlan = RoutePlanSearch. newInstance( ) ; 
myRoutePlan. setOnGetRoutePlanResultListener( 
new OnGetRoutePlanResultListener( ){ 


(Override 
public void onGetWalkingRouteResult(WalkingRouteResult walkingRouteResult){ } 
(Override 
public void onGetTransitRouteResult(TransitRouteResult transitRouteResult){ } 
(Override 
public void onGetMassTransitRouteResult( 
MassTransitRouteResult massTransitRouteResult){ } 
(@ Override 
public void onGetDrivingRouteResult(DrivingRouteResult result){ 
if(result. getRouteLines() == nul1){ 
Toast. makeText(MainActivity. this, 
" 抱 表 ,未 搜索 到 驾车 路 线 !" ,Toast. LENGTH_SHORT). show( ) ; 
}elsef 
myBaiduMap. clear( ); 
MyRouteOverlay myRouteOverlay = new MyRouteOverlay(myBaiduMap); 


// 将 路 线 数据 传递 至 图 层 
myRouteOverlay. setData(result. getRouteLines( ).get(0)); 
myRouteOverlay. addToMap( ); // 将 路 线 图 显示 在 地 图 上 
myRouteOverlay. zoomToSpan( ); // 自 适应 缩放 
// 设 置 路 线 节 点 点 击 监听 
myBaiduMap. setOnMarkerClickListener(myRouteOverlay); 
} } 
(Override 
public void onGetIndoorRouteResult( IndoorRouteResult indoorRouteResult){ } 
(Override 


public void onGetBikingRouteResult(BikingRouteResult bikingRouteResult){ } 
}); 
myStartNode = PlanNode.withLocation(new LatLng( 
Double. parseDouble(myEditPointl1A. getText().toString( )), 
Double. parseDouble(myEditPoint1B. getText().toString( ) ) ) ) ; 
myEndNode = PlanNode. withLocation(new LatLng( 
Double. parseDouble(myEditPoint2A. getText( ) .toString( ) ) ， 
Double. parseDouble(myEditPoint2B. getText( ) .toString( ) ) ) ) ; 
// 开 始 搜索 最 佳 驾车 路 线 , 并 将 路 线 显示 在 地 图 上 
myRoutePlan. drivingSearch( ( 
new DrivingRoutePlan0ption( ) ) .Erom(myStartNode) .to(myEndNode) ); 


myRoutePlan. destroy( ) ; // 销 毁 搜 索 对 象 ,释放 内 存 
} 
public void onClickBtn2(View v) { // 响 应 单 击 "分 享 驾车 线路 "按钮 
// 创 建 分 享 检索 实例 ; 
ShareUrlSearch myShareUrlSearch = ShareUrlSearch. newInstance( ); 
// 创 建设 置 分 享 检索 监听 者 ; 


myShareUrlSearch. setOnGetShareUrlResultListener( 
new OnGetShareUrlResultListener() { 
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(Override 

public void onGetPoiDetailShareUrlResult(ShareUrlResult shareUrlResult){ } 
(Override 

public void onGetLocationShareUrlResult(ShareUrlResult shareUrlResult){ } 
(WOverride 

// 使 用 Intent 执行 分 享 驾车 线路 动作 


public void onGetRouteShareUrlResult(ShareUrlResult shareUrlResult) { 
Intent myIntent = new Intent(Intent.ACTION SEND); 
myIntent. putExtra( Intent. EXTRA TEXT, 
"通过 百度 地 图 分 享 的 驾车 线路 是 : ”+ shareUrlResult. getUr1l() ); 
myIntent. setType(" text/plain" ) ; 
startActivity(Intent. createChooser(myIntent, "百度 地 图 驾车 线路 分 享 " ) ); 
} }); 
RouteShareURLOption. RouteShareMode myRouteShareMode = 
RouteShareURLOption. RouteShareMode. CAR ROUTE SHARE MODE; 
myShareUrlSearch. requestRouteShareUr1(new RouteShareURLOption( ) 
.from(myStartNode).to(myEndNode).routMode(myRouteShareMode) ); 
myShareUrlSearch. destroy( ); // 销 毁 分 享 检索 实例 ; 
} 
class MyRouteOverlay extends DrivingRouteOverlay!{ 
public MyRouteOverlay(BaiduMap baiduMap) {super(baiduMap);} 
(Override 
public boolean onRouteNodeClick( int i){ // 通 过 Toast 显示 驾车 路 线 建议 信息 
Toast. makeText (MainActivity. this, "驾车 建议 :" + myRouteLine. getAllStep(). get(i).getInstructions( )， 
Toast. LENGTH SHORT). show( ); 
return true; 


时 


上 面 这 段 代 码 在 MyCode\ MySampleX65\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX65\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引信 百度 地 图 组 件 , 即 MyCode\MySampleX65\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySampleX65\app\build. gradle 文件 中 添加 compile files ('libs/BaiduLBS_Android. jar') 和 compile 
' com。 google. android. gms: play-services-appindexing: 8. 4. 0' 依 赖 项 。 此 外 ,还 要 在 
AndroidManifest. xml 文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 
开发 说 明 请 参考 官方 网 址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 目 在 MyCodeA\ 
MySampleX65 文件 夹 中 ， 


281 使 用 百度 SDK 调用 百度 地 图 App 的 驾车 导航 


此 实例 主要 通过 使 用 百度 地 图 SDK 的 BaiduMapNavigation ,实现 根据 起 点 和 终点 对 驾车 线路 进 
行 导航 。 当 实例 运行 之 后 ,在 “起 点 纬度 : ”输入 框 中 输入 起 点 纬度 值 , 如 “29.733082”, 在 “起 点 经 度 :” 
输入 框 中 输入 起 点 经 度 值 , 如 “106. 590041”, 在 “终点 纬度 : "输入 框 中 输入 终点 纬度 值 , 如 
“29. 919309”, 在 “终点 经 度 : ”输入 框 中 输入 终点 经 度 值 ,如 “107. 248061” ,如 图 281. 1 的 左 图 所 示 ; 
然后 单 击 “ 调 用 百度 地 图 的 驾车 导航 功能 ”按钮 , 则 将 启动 百度 地 图 的 驾车 导航 功能 ,如 图 281. 1 的 右 


第 9 章 ”第 三 方 SDK 开 发 


图 所 示 。 注 意 : 
中 重 


MySample 


起 点 纬度 : 9.733082 


起 点 经 度 : 106.590041 


终点 纬度 : 29.919309 


终点 经 度 : 107.248061 


调用 百度 地 图 的 驾车 导航 功能 


剩余 100 公 里 1 小 时 28 分 钟 
12:14 到 达 更 多 


主要 代码 如 下 : 
public void onClickBtnl(View v) { // 响 应 单 击 "调用 百度 地 图 的 驾车 导航 功能 "按钮 
LatLng myPointl1 = 


new LatLng(Double. valueOf (myEditPoint1A. getText().toString()), 
Double. valueOf (myEditPoint1B. getText().toString( ))); 
LatLng myPoint2 = 
new LatLng(Double. valueOf (myEditPoint2A. getText().toString()), 
Double. valueOf (myEditPoint2B. getText().toString( ))); 
NaviPara0ption myNaviPara0ption = new NaviParaOption() 
. StartPoint(myPoint1l ) .endPoint(myPoint2 ) 
. startName(" 我 的 起 点 " ) .endName(" 我 的 终点 " ); 
try {// 调 用 百度 地 图 的 驾车 导航 功能 
BaiduMapNavigation. openBaiduMapNavi(myNaviParaOption, this); 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e. getMessage( ) .toString( ) ，Toast. LENGTH SHORT). show( ); 
} } 


上 和 面 这 段 代 码 在 MyCode\ MySampleX55\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 外 ,此 实例 需要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX55\ app\\ 
libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\MySampleX55\app\build. gradle 文件 中 添加 compile files 
('libs/BaiduLBS Android. jar') 和 compile 'com. google. android. gms: play 一 services 一 appindexing : 
8. 4.0' 依 赖 项 。 并 且 还 要 在 AndroidManifest. xml 文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 
直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 址 : http://lbsyun. baidu. com/sdk。 此 实例 的 完 
整 项 日 在 MyCode\MySampleX55 文件 夹 中 ， 
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282 ”使 用 百度 SDK 调用 百度 地 图 App 的 POI 检索 


此 实例 主要 通过 使 用 百度 地 图 SDK 的 BaiduMapPoiSearch ,实现 根据 指定 的 经 度 、 纬度、 搜索 半 
径 及 名 称 调用 百度 地 图 的 POI 周边 检索 页 面 。 当 实例 运行 之 后 ,在 “纬度 : "输入 框 中 输入 重庆 园 博 
园 的 纬度 ,如 "29. 680962”, 在 “经 度 : ”输入 框 中 输入 重庆 园 博 园 的 经 度 ,如 "106. 570132”, 在 “搜索 半 
径 ( 米 ) : ”输入 框 中 输入 搜索 半径 ,如 “2000”, 在 “名 称 : ”输入 框 中 输入 “重庆 园 博 园 ”, 如 图 282. 1 的 
左 图 所 示 ,然后 单 击 "启动 百度 地 图 POI 周边 检索 页 面 ” 按 钮 , 则 将 启动 百度 地 图 POI 周边 检索 ,并 显 
示 搜 索 结 果 , 如 图 282. 1 的 右 图 所 示 。 注 意 : 测试 手机 需要 安装 百度 地 图 App。 


中 香 D 束 “ 0o% 上 午 8:48 | 
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1.2km | 重庆 市 渝 北 区 龙 景 路 1 号 
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图 282.1 
主要 代码 如 下 : 
public void onClickBtnl (View v) { // 响 应 单 击 " 启 动 百度 地 图 POI 周边 检索 页 面 "按钮 
LatLng myPoint = 


new LatLng(Double. valueOf (myEditPointA. getText().toString( )), 
Double. valueOf (myEditPointB. getText().toString())); 
PoiPara0ption myPoiPara0ption = new PoiParaOption() 
.key(myEditName. getText( ) .toString( ) ) .center(myPoint) 
.radius(Integer.parseInt(myEditRadius. getText( ) .toString() ) ); 
try { // 调 启 百 度 地 图 poi 周边 检索 页 面 
BaiduMapPoiSearch. openBaiduMapPoiNearbySearch(myPoiParaOption, this); 
} catch (Exception e) { 
Toast. makeText (MainActivity. this, 
e.getMessage( ). toString(), Toast.LENGTH SHORT). show( ) ; 


} } 


上 面 这 段 代 码 在 MyCode\ MySampleX60\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 引入 百度 地 图 组 件 , 即 MyCode\ 
MySampleX60\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\MySampleX60\app\build. gradle 文件 
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中 添加 compile files('libs/BaiduLBS_Android. jar') 和 compile 'com. google. android. gms: play 一 
services 一 appindexing:8.4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文件 中 添加 开发 密 钥 和 操作 权限 
(参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 址 : http://lbsyun. baidu. com/ 
sdk。 此 实例 的 完整 项 目 在 MyCode\MySampleX60 文件 夹 中 。 


283 ”使 用 百度 SDK 实现 在 地 图 中 定位 手机 位 置 


此 实例 主要 通过 使 用 百度 地 图 SDK ,实现 在 百度 地 图 中 定位 当前 手机 位 置 。 当 实例 运行 之 后 , 单 
击 “ 在 百度 地 图 中 定位 当前 手机 位 置 ” 按 钮 , 则 在 百度 地 图 中 显示 以 当前 手机 位 置 为 中 心 的 地 图 ,效果 
如 图 283. 1 所 示 。 


@ rn ‘Dr vw dB 56% 17:56 
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图 283.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 

MapView myMapView; 

BaiduMap myBaiduMap; 

(Override 

protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
SDKInitializer. initialize(getApplicationContext( )); 
setContentView(R. layout. activity_main); 
myMapView = (MapView) findViewById(R. id.myMapView); 
myBaiduMap = myMapView.getMap(); 
MapStatusUpdate myMapStatusUpdate = 

MapStatusUpdateFactory. zoomTo( 20); // 放 大 显示 当前 地 图 

myBaiduMap. setMapStatus(myMapStatusUpdate); 
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public void onClickBtnl(View v){ // 响 应 单 击 "在 百度 地 图 中 定位 当前 手 
机 位 置 "按钮 
myBaiduMap. setMyLocationEnabled(true); // 启 用 定位 图 层 


LocationClient myLocationClient = new LocationClient(this); 
myLocationClient. registerLocationListener(new BDLocationListener(){ 
(@ Override 
public void onReceiveLocation( BDLocation location){ // 获 取 定 位 结果 
if (location == null | | myMapView == null){ 
Toast. makeText(MainActivity. this, 
"未 获取 到 当前 位 置 ,请 稍 后 重 试 !", Toast. LENGTH_SHORT). show( ); 


return; 
}else{ 
LatLng myLocation = 
new LatLng( location. getLatitude( ), location. getLongitude( ) ); 
MapStatusUpdate myMapStatusUpdate = 
MapStatusUpdateFactory. newLatLng( myLocation); 
myBaiduMap. setMapStatus (myMapStatusUpdate); // 将 地 图 中 心 设置 为 当前 位 置 


MyLocationData myLocationData = new MYLocationData. Builder() 
.accuracy( location. getRadius()).1latitude( location. getLatitude( )) 
.longitude( location. getLongitude( ) ) .build(); 

myBaiduMap. setMyLocationData(myLocationData); // 设 置 定位 数据 


// 在 地 图 上 显示 当前 位 置 

myBaiduMap. setMyLocationConfigeration(new MYLocationConfiguration( 
MyLocationConf iguration. LocationMode. NORMAL, true, null)); 

Toast. makeText(MainActivity.this, "定位 成 功 !",Toast.LENGTH SHORT). show(); 


} } }); 
LocationClientOption myLocationClientOption = new LocationClientOption( ); 
myLocationClientOption. setOpenGps(true); // 打 开 GPS 功能 
myLocationClientOption. setCoorType(" bd0911" ); // 设 置 坐标 类 型 
myLocationClient. setLocOption(myLocationClientOption); // 设 置 定位 参数 
myLocationClient. start( ); // 开 始 定位 
}} 


上 面 这 上段 代码 在 MyCode\ MySample921\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySample921\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 需 要 说 明 的 是 ， 
此 实例 需要 引入 百度 地 图 组 件 , 即 MyCode\MySample921\app\libs 目录 下 的 所 有 内 容 , 然 后 在 
MyCode\ MySample921\app\build. gradle 文件 中 添加 compile files ('libs/BaiduLBS_Android. jar" 和 
compile 'com. google. android. gms:play-services-appindexing:8. 4.0' 依 赖 项 ,并 在 AndroidManifest. 
xml 文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代 码 )。 百 度 地 图 开发 说 明 请 参考 官 
方 网 址 : http://lbsyun. baidu. com/sdk。 此 实例 的 完整 项 目 在 MyCode\MySample921 文件 夹 中 ， 


284 使 用 百度 SDK 获取 在 地 图 上 点 击 位 置 的 地 名 


此 实例 主要 通过 使 用 百度 地 图 SDK ,实现 在 单 击 百度 地 图 时 显示 该 点 的 位 置信 息 。 当 实例 运行 
之 后 ,在 百度 地 图 中 任意 点 击 ( 主 干线 ) , 则 在 弹出 的 Toast 中 显示 该 单 击 位 置 的 具体 地 址 信息 ,效果 
分 别 如 图 284. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 
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public class MainActivity extends Activity { 
(QOverride 
protected void onCreate( Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
SDKInitializer. initialize(getApplicationContext( )); 
setContentView(R. layout.activity main); 
MapView myMapView = (MapView) findViewById(R. id.MyMapView); 
BaiduMap myBaiduMap = myMapView. getMap( ); 
LatLng myLatLng = new LatLng(29.6876, 106.5979); 
MapStatus myMapStatus = 
new MapStatus. Builder().target(myLatLng).zoom(15).build( ); 
myBaiduMap. setMapStatus( MapStatusUpdateFactory. newMapStatus(myMapStatus) ); 
// 点 击 地 图 监听 
myBaiduMap. setOnMapClickListener(new BaiduMap. OnMapClickListener() { 
@ Override 
public void onMapClick(LatLng latLng) { 
GeoCoder myGeoCoder = GeoCoder.newInstancel( ); 
myGeoCoder. setOnGetGeoCodeResultListener( 
new OnGetGeoCoderResultListener() { 

@Override 

public void onGetGeoCodeResult(GeoCodeResult geoCodeResult) { } 

(Override 

// 坐 标 转换 为 地 址 的 回调 函数 

public void onGetReverseGeoCodeResult( 

ReverseGeoCodeResult reverseGeoCodeResult) { 
Toast. makeText(MainActivity.this, "刚才 点 击 的 位 置 是 :”+ 
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reverseGeoCodeResult. getAddress(), Toast.LENGTH SHORT). show( ); 
} }); 
// 将 点 击 地 点 坐标 转换 为 地 址 信息 
myGeoCoder. reverseGeoCode(new ReverseGeoCodeOption(). location( latLng)); 


myGeoCoder. destroy( ); 
} 
(Override 
public boolean onMapPoiClick(MapPoi mapPoi) { return false; } 
和 


上 和 面 这 上段 代码 在 MyCode\ MySample915\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySample915\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySample915\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySample915\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile ， 
com. google. android. gms:play 一 services 一 appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 
文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. comy/sdk。 此 实例 的 完整 项 目 在 MyCode\MySample915 文件 夹 中 ， 


285 使 用 百度 SDK 在 地 图 的 城市 之 间 绘 制 连 线 


此 实例 主要 通过 使 用 百度 SDK 在 百度 地 图 中 根据 两 个 城市 的 经 纬度 值 实现 连 线 的 效果 。 当 实 
例 运 行 之 后 , 单 击 “ 显 示 四 点 连 线 效果 ”按钮 , 则 将 用 直线 将 西安 、 郑 州 、 长 沙 、 重 庆 连 接 起 来 ,形成 一 个 
封闭 的 四 边 形 ,效果 分 别 如 图 285. 1 的 左 图 和 右 图 所 示 。 
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图 285. 1 
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public void onClickBtnl(View v) { // 响 应 单 击 按钮 "显示 四 点 连 线 效果 " 
LatLng myPoint1 = new LatLng(34.338739,108.896178); // 西 安 的 纬度 、 经 度 
LatLng myPoint2 = new LatLng(34.764961,113.605886); // 郑 州 的 纬度 、 经 度 
LatLng myPoint3 = new LatLng(28.245249,112.943583); // 长 沙 的 纬度 、 经 度 
LatLng myPoint4 = new LatLng(29.557184,106.552688); // 重 庆 的 纬度 、 经 度 
myPoints = new ArrayList <LatLng >(); 
myPoints.add(myPoint]1 ); 
myPoints. add(myPoint2); 
myPoints.add(myPoint3); 
myPoints.add(myPoint4); 
myPoints.add(myPoint]1 ); 
OverlayOptions myPolyline = new PolylineOptions().width(10). 

color(0xRRFF0000). points(myPoints); // 绘 制 折 线 ( 连 线 ) 

myBaiduMap. addOverlay(myPolyline); 


主要 代码 如 下 : 


上 面 这 段 代 码 在 MyCode\ MySampleX10\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleXl0\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleXl0\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySampleXl10\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms: play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 298 或 直接 看 源 代 码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 目 在 MyCode\MySampleX10 文件 夹 中 ， 
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此 实例 主要 通过 使 用 百度 地 图 SDK 中 的 InfoWindow, 实 现在 折 度 地 图 的 指定 位 置 添加 图 文 悬 
浮 框 。 当 实例 运行 之 后 , 单 击 " 在 指定 的 位 置 添 加 图 文 悬 浮 框 ?按钮 , 则 在 指定 的 位 置 (武汉 的 上 方 ) 添 
加 一 个 图 文 悬浮 框 ,效果 分 别 如 图 286. 1 的 左 图 和 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickBtnl(View v) { // 响 应 单 击 "在 指定 的 位 置 添加 图 文 悬浮 框 " 按 钮 
LatLng myPoint = new LatLng(30.53553,114.324234);  ”// 武 汉 的 纬度 经度 
TextView myTextView = new TextView(getApplicationContext()); 
NinePatchDrawable myNinePatchDrawable = (NinePatchDrawable) 
getDrawable(R. drawable. luobackground); // 使 用 指定 图 设置 图 文 悬 浮 框 的 背景 
myTextView. setBackground(myNinePatchDrawable); 
Drawable myDrawable = getResources().getDrawable(R.mipmap.myimagel ); 
// 在 调用 setCompoundDrawables( ) 方 法 时 ， 
// 必 须 首 先 调 用 Drawable. setBounds() 方 法 ,否则 图 像 不 显示 
myDrawable. setBounds( - 5, 0, 
myDrawable. getMinimumWidth(), myDrawable. getMinimumHeight( )); 
// 设 置 图 像 顺序 是 : 左 、 上 、 右 \ 下 ,此 代码 仅 在 文字 左边 显示 图 像 
myTextView. setCompoundDrawables(myDrawable, null, null, null); 
myTextView. setWidth(780); // 设 置 图 文 悬 浮 框 宽度 
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myTextView. setHeight( 430); // 设 置 图 文 悬 浮 框 高 度 
myTextView. setTextColor( Color. BLUE); // 设 置 文字 颜色 
myTextView. setPadding(20, 10, 10, 10); // 设 置 内 边 距 ,顺序 是 : 左 、 上 、 右 \ 下 


myTextView. setText(" 武汉 地 处 江汉 平原 东部 ,长 江 及 其 最 大 支流 汉 水 横贯 市 境 中 央 , 将 武汉 城区 一 分 为 三 ， 
形成 了 武昌 、 汉 口 .汉阳 三 镇 隔 江 鼎立 的 格局 ."); 

// 在 InfoWindow 中 加 载 图 文 悬 浮 框 (在 纵向 偏 移 - 60, 以 露出 武汉 2 字 ) 

InfoWindow myInfoWindow = new InfoWindow(myTextView, myPoint, - 60); 

myBaiduMap. showInfoWindow( myInfoWindow); // 显 示 InfoWindow 
} 
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上 面 这 段 代 码 在 MyCode\ MySampleX20\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX20\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX20\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode'\ 
MySampleX20\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms: play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 铀 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 日 在 MyCode\MySampleX20 文件 夹 中 ， 


287 ”使 用 百度 SDK 在 地 图 上 添加 淡 人 和 人 动 轩 


此 实例 主要 通过 使 用 百度 地 图 SDK 中 的 InfoWindow, 实 现在 百度 地 图 的 指定 位 置 添加 淡 入 动 
画 。 当 实例 运行 之 后 , 单 击 * 在 指定 的 位 置 添 加 淡 和 动画 ?按钮 , 则 在 指定 的 位 置 (重庆 江北 嘴 金 融 城 ) 
浮 出 一 个 由 透明 变 为 不 透明 的 建筑 物 图像 ,效果 分 别 如 图 287. 1 的 左 图 和 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickBtni (View v) { // 响 应 单 击 "在 指定 的 位 置 添加 淡 入 动画 "按钮 
// 重 庆 江 北 嘴 金 融 城 的 纬度 .经度 
LatLng myPoint = new LatLng(29.580759,106.576481); 
ImageView myImageView = new ImageView(getApplicationContext( )); 
myImageView. setImageResource(R.mipmap.myimagel ); 
// 创 建 淡 入 图 像 动 画 (0f 表示 完全 透明 ,1.0f 表示 完全 不 透明 ) 
AlphaAnimation myAnimation = new AlphaAnimation(0f, 1.0f); 
// 设 置 动画 持续 时 间 5 秒 
myAnimation. setDuration(5000 ) ; 
// 在 myImageView 上 添加 淡 入 动画 myAnimation 
myImageView. startAnimation(myAnimation); 
// 创 建 线 性 布局 管理 器 
LinearLayout myLinearLayout = new LinearLayout(this); 
// 以 垂直 方式 布局 界面 控件 
myLinearLayout. setOrientation(LinearLayout. VERTICRL ) ; 
// 在 线性 布局 管理 器 中 添加 myImageView 控件 
myLinearLayout. addView( myImageView); 
// 在 InfoWindow 中 加 载 myLinearLayout, 实现 淡 入 图 像 动 
InfoWindow myInfoWindow = new InfoWindow(myLinearLayout, myPoint,160); 
// 显 示 InfoWindow 
myBaiduMap. showInfoWindow(myInfoWindow); 


上 面 这 段 代 码 在 MyCode\ MySampleX22\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX22\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX22\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySampleX22\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
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'com. google. android. gms: play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 铀 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/sdk。 此 实例 的 完整 项 目 在 MyCode\MySampleX22 文件 夹 中 ， 


288 使 用 百度 SDK 在 地 图 上 添加 弹跳 型 动画 


此 实例 主要 通过 使 用 百度 地 图 SDK 中 的 MarkerOptions, 实 现在 百度 地 图 的 指定 位 置 添 加 弹跳 
型 动画 。 当 实例 运行 之 后 , 单 击 "在 指定 的 位 置 添加 弹跳 型 动画 ?按钮 , 则 篮球 将 在 指定 的 位 置 (重庆 
奥 林 匹 殉 体 育 中 心 ) 上 下 弹跳 ,效果 分 别 如 图 288. 1 的 左 图 和 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickBtn1 (View v) { // 响 应 单 击 "在 指定 的 位 置 添加 弹跳 型 动画 "按钮 
myBaiduMap. clear( ); 
// 定 义 动画 图 像 的 (重庆 奥林匹克 体育 中 心 ) 坐 标点 (纬度 经度) 
LatLng myPoint = new LatLng(29.531138,106.512525); 
BitmapDescriptor myBitmap = BitmapDescriptorFactory. 
fromResource(R. mipmap. myimagel ); // 获 取 图 像 资源 
MarkerOptions myMarkerOptions = new MarkerOptions().position(myPoint). 
icon(myBitmap).zIndex(0).period(10); // 根 据 图 像 创建 MarkerOptions 
// 在 MarkerOptions 中 设置 弹跳 型 动画 MarkerAnimateType. jump 
myMarkerOptions. animateType( MarkerOptions. MarkerAnimateType. jump); 
myBaiduMap. addOver lay( myMarkerOptions); // 在 百度 地 图 中 添加 弹跳 型 动画 
} 


上 和 面 这 段 代 码 在 MyCode\ MySampleX33\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX33\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX33\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCodeA\ 
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MySampleX33\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms: play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 上 日 在 MyCode\MySampleX33 文件 夹 中 ， 


289 使 用 百度 SDK 在 地 图 上 查询 指示 城市 兴趣 扣 


此 实例 主要 通过 使 用 百度 地 图 SDK, 实 现在 百度 地 图 中 查询 指定 城市 的 宾馆 、 美 食 等 兴趣 点 
(POD 信 息 。 当 实例 运行 之 后 ,在 “城市 : "输入 框 中 输入 城市 名 称 ,如 “重庆 ”, 在 “兴趣 点 : ”输入 框 中 
输入 "火锅 ”, 然 后 单 击 “ 开 始 搜索 ”按钮 , 则 在 地 图 中 显示 重庆 的 火锅 店 分 布 图 , 单 击 其 中 悬浮 的 任意 
一 个 图 标 , 则 在 弹出 的 Toast 中 显示 该 火锅 店 的 名 称 , 如 图 289. 1 的 左 图 所 示 。 如 果 在 “城市 : ”输入 
框 中 输入 “ 渝 北 ”, 在 “兴趣 点 : ”输入 框 中 输入 "楼盘 ,然后 单 击 “ 开 始 搜索 ”按钮 , 则 在 地 图 中 显示 渝 北 
的 楼 盘 分 布 图 , 单 击 其 中 悬浮 的 任意 一 个 图 标 , 则 在 弹出 的 Toast 中 显示 该 楼 盘 的 名 称 , 如 图 289. 1 
的 右 图 所 示 。 
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主要 代码 如 下 : 


public void onClickBtnl(View v) { // 响 应 单 击 " 开 始 搜索 "按钮 
myPoiSearch = PoiSearch. newInstance( ); 
myPoiSearch. setOnGetPoiSearchResultListener( 
new OnGetPoiSearchResultListener( ){ 
(Override 
public void onGetPoiResult(PoiResult result) { 
if (result.error == SearchResult. ERRORNO. NO_ERROR) { 
myBaiduMap. clear(); // 重 置地 图 
MarkerOverlay myOverlay = new MarkerOverlay(myBaiduMap); 
myBaiduMap. setOnMarkerClickListener(myOverlay); 
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myOverlay. setData( result); // 传 入 目标 兴趣 点 相关 数据 
myOverlay. addToMap( ); // 在 地 图 上 显示 
myOverlay. zoomToSpan( ); // 自 适应 缩放 

} else { 


Toast. makeText (MainActivity. this, 
"暂时 未 搜索 到 相关 " + myEditTarget, Toast.LENGTH SHORT). show( ); 


} } 

(@ Override 

public void onGetPoiDetailResult(PoiDetailResult poiDetailResult) {} 
(QW Override 


public void onGetPoiIndoorResult(PoiIndoorResult poiIndoorResult) { } }); 
myPoiSearch. searchInCity( (new PoiCitySearchOption( )) 
.City(myEditCity. getText().toString( )) 
keyword(myEditTarget. getText().toString( )) 
.pageNum(10)); // 在 指定 城市 搜索 目标 ,最 多 显示 10 个 
} 
class MarkerOverlay extends PoiOverlay { 
public MarkerOverlay(BaiduMap baiduMap) { super(baiduMap); } 
(OOverride 
// 自 定义 标记 点 击 监听 事件 
public boolean onPoiClick(int i) { 
// 获 取 点 击 位 置 对 应 的 信息 
PoiInfo myPoiInfo = getPoiResult().getAllPoi().get(i); 
Toast. makeText (MainActivity. this, myPoiInfo. name, Toast. LENGTH SHORT). show( ) ; 


return true; 


3 


上 面 这 段 代 码 在 MyCode\ MySampleX41\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX41\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX41\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySampleX41\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms: play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. comy/sdk。 此 实例 的 完整 项 目 在 MyCode\MySampleX41 文件 夹 中 ， 


290 ”使 用 百度 SDK 在 地 图 上 为 行政 区 还 加 边 界 线 


此 实例 主要 通过 使 用 百度 地 图 SDK 的 DistrictSearch ,实现 为 指定 行政 区 添加 自 定 义 边界 线 。 当 
实例 运行 之 后 ,在 “重庆 市 的 区 县 名 称 : “输入 框 中 输入 区 县 名 称 , 如 “ 巴 南 区 ”, 然 后 单 击 “ 开 始 查 询 ” 按 
钮 , 则 在 百度 地 图 中 使 用 红色 的 虚线 标注 巴 南 区 的 行政 区 域 ,如 图 290. 1 的 左 图 所 示 。 如 果 在 “重庆 
市 的 区 县 名 称 : ”输入 框 中 输入 “北碚 区 ”, 然 后 单 击 “ 开 始 查 询 ” 按 钮 , 则 在 百度 地 图 中 使 用 红色 的 虚线 
标注 北碚 区 的 行政 区 域 ,如 图 290. 1 的 右 图 所 示 。 
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MySample 


重庆 市 的 区 县 名 称 : 巴 南 区 开始 查询 i 重庆 市 的 区 县 名 称 : 比 碚 区 开始 查询 


主要 代码 如 下 : 


// 响 应 单 击 "开始 查询 "按钮 
public void onClickBtnl (View v) { 
myBaiduMap. clear( ); 
// 建 立 查 询 条 件 
DistrictSearchOption myDistrictSearchOption = new DistrictSearchOption( ) . 
cityName(" 重庆 ").districtName(myEditText.getText().toString()); 
// 设 置 查询 监听 事件 
myDistrictSearch. setOnDistrictSearchListener( 
new OnGetDistricSearchResultListener() { 
@ Override 
public void onGetDistrictResult(DistrictResult districtResult) { 
if (districtResult.error == SearchResult. ERRORNO. NO ERROR) { 
List<List< LatLng >> pointsList = districtResult. getPolylines(); 
if (pointsList == null) return; 
for (List <LatLng> polyline : pointsList) { 
OverlayOptions myPolylineOptions = new PolylineOptions().width(10) 
.Points(polyline).dottedLine(true).color(Color. RED); 
// 添 加 区 域 边界 线 Overlay 
myBaiduMap. addOverlay(myPolylineOptions); 
yy} 
// 执 行 行政 区 域 的 查询 
myDistrictSearch. searchDistrict(myDistrictSearchOption); 
} 


上 和 面 这 段 代 码 在 MyCode\ MySampleX44\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX44\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX44\app\libs 目录 下 的 所 有 内 容 ,然后 在 MyCode\ 
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MySampleX44\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms: play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 目 在 MyCode\MySampleX44 文件 夹 中 。 


291 使 用 百度 SDK 在 地 图 指定 范围 添加 圆 角 和 定形 


此 实例 主要 通过 使 用 百度 地 图 SDK 中 的 GroundOverlayOptions ,实现 在 百度 地 图 指定 的 地 理 范 
围 内 添加 线性 渐变 的 圆 角 和 矩形 。 当 实例 运行 之 后 , 单 击 " 在 指定 的 地 理 范 围 添 加 圆 角 矩形 ”按钮 , 则 在 
指定 的 矩形 范围 (重庆 两 江 新 区 ) 内 显示 一 个 线性 渐变 的 圆 角 矩形 及 文字 "重庆 两 江 新 区 ”, 效 果 分 别 
如 图 291. 1 的 左 图 和 右 图 所 示 。 


MySample MySample 
在 指定 的 地 理 范围 添加 圆 角 矩形 在 指定 的 地 理 范围 添加 圆 角 和 矩形 
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主要 代码 如 下 : 
public void onClickBtnl(View v) { // 响 应 单 击 "在 指定 的 地 理 范 围 添 加 圆 角 和 矩形" 
// 按 钮 
myBaiduMap. clear( ); 
// 定 义 圆 角 矩形 的 地 理 范 围 
LatLng myPoint1l = new LatLng(29.810734,106.405711);  // 左 上 角 纬 度 经 度 
LatLng myPoint2 = new LatLng(29.517988,106.949006);  // 右 下 角 纬 度 经 度 


LatLngBounds myBounds = new LatLngBounds. Builder() 

.include(myPointl ). include(myPoint2).build( ); 
TextView myTextView = new TextView(getApplicationContext()); 
myTextView. setWidth(250); 
myTextView. setHeight(250); 
myTextView. setText(" 重 庆 两 江 新 区 " ); 
myTextView. setGravity(Gravity. CENTER); // 设 置 文字 居中 对 齐 
myTextView. setTextSize( 28); // 设 置 文字 大 小 
myTextView. setTextColor( Color. RED); // 设 置 文字 颜色 


} 
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// 在 TextView 上 设置 圆 角 和 矩形 背景 

myTextView. setBackground( getDrawable(R. drawable. myshape) ); 

// 创 建 线性 布局 管理 器 

LinearLayout myLinearLayout = new LinearLayout(this); 

// 以 垂直 方式 布局 界面 控件 

myLinearLayout. setOrientation(LinearLayout.VERTICRL ) ; 

// 在 线性 布局 管理 器 中 添加 myTextView 控件 

myLinearLayout. addView(myTextView); 

// 将 线性 布局 转换 为 BitmapDescriptor 

BitmapDescriptor myShape = BitmapDescriptorFactory. fromView(myLinearLayout); 
// 根 据 地 理 范 围 和 圆 角 矩形 创建 GroundOverlayOptions 

GroundOverlayOptions myGroundOverlayOptions = new GroundOver1layOptions( ) 


. PositionFromBounds( myBounds) // 设 置地 理 范围 
. image( myShape) // 设 置 覆 盖 图 形 
:transparency(0.7f£); // 设 置 图 像 透 明度 


myBaiduMap. addOverlay(myGroundOverlayOptions); 
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上 和 面 这 段 代 码 在 MyCode\ MySampleX36\app\src\main\java\com\bin\luo\mysample\ 


MainActivity. java 文件 中 。 在 这 段 代 码 中 ,myTextView. setBackground(getDrawable (R. drawable. 
myshape) ) 表 示 使 用 XML 文件 myshape 配置 的 线性 渐变 的 圆 角 和 矩形 设置 TextView 控件 的 背景 ， 
myshape 文件 的 主要 内 容 如 下 : 


<?xml] version = "1.0" encoding = "utf 一 8"?> 
< shape xmlns:android = "http://schemas. android. com/apk/res/android" 


android: shape = " rectangle"> 


< size android:width = "250dp" android:height = "250dp"/> 


< stroke android:width = "2dp" 
android: color = " # ff0000" 
android: dashGap = " 4sp" 
android: dashWidth = "4dp" /> 


< gradient android:angle = "270" 


android: endColor = " #0000ff" 
android: startColor = " #ffffff" /> 


< corners android: radius = "20dp" /> 


</shape> 


上 面 这 段 代 码 在 MyCode\MySampleX36\app\src\main\res\drawable\myshape. xml 文件 中 。 


注意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 MyCode\MySampleX36\ app\src\ 
main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需要 引入 百度 地 图 组 件 , 即 MyCode\ 
MySampleX36\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\MySampleX36\ app\build. gradle 文件 
中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 'com. google. android. gms: play- 
services-appindexing:8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文件 中 添加 开发 密 钥 和 操作 权限 ( 参 
考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 址 : http://lbsyun. baidu. com/ sdk。 
此 实例 的 完整 项 目 在 MyCode\MySampleX36 文件 夹 中 。 


292 ”使 用 百度 SDK 查询 指定 地 点 的 热力 图 


此 实例 主要 通过 使 用 百度 地 图 SDK ,实现 查询 指定 经 度 和 纬度 的 热力 图 。 当 实例 运行 之 后 ,在 
“纬度 : ”输入 框 中 输入 “29. 6876”, 在 “经 度 : ”输入 框 中 输入 “106. 5979”, 然 后 单 击 “ 显 示 热 力图 ” 按 
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钮 , 则 在 百度 地 图 中 显示 重庆 主 城 的 热力 图 ,效果 如 图 292. 1 的 左 图 所 示 ; 如 果 在 “纬度 : ”输入 框 中 
输入 “30. 67”, 在 “经 度 : ”输入 框 中 输入 “104. 06”, 然 后 单 击 “显示 热力 图 ”按钮 , 则 在 百度 地 图 中 显示 
成 都 主 城 的 热力 图 ,效果 如 图 292. 1 的 右 图 所 示 。 


MySample MySample 


纬度 : 9.6876 经 度 : 106.5979 纬度 : 30.67 经 度 : 104.06 


显示 热力 图 显示 热力 图 


主要 代码 如 下 : 

public void onClickBtn1 (View v) { // 响 应 单 击 " 显 示 热 力图 "按钮 
myBaiduMap = myMapView. getMap(); 
LatLng myLocation = 


new LatLng(Double. parseDouble(myEditLatitude. getText().toString( )), 
Double. parseDouble( myEditLongitude. getText( ).toString( ))); 
MapStatus myMapStatus = new MapStatus.Builder().target(myLocation).build( ); 
myBaiduMap. setBaiduHeatMapEnabled( true); // 人 允许 显示 热力 图 层 
} 


上 面 这 段 代 码 在 MyCode\ MySample916\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySample916\ app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySample916\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySample916\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_Android. jar') 和 compile ， 
com. google. android. gms:play 一 services 一 appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 
文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 目 在 MyCode\MySample916 文件 夹 中 。 


293 ”使 用 百度 SDK 实现 隐藏 或 显示 地 名 标注 信息 


此 实例 主要 通过 使 用 百度 地 图 SDK 的 showMapPoi() 方 法 ,实现 隐藏 或 显示 百度 地 图 的 地 名 标 
注 等 信息 。 当 实例 运行 之 后 , 单 击 “ 显 示 标 注 信 息 ” 按 钮 , 则 在 百度 地 图 中 显示 每 个 POI 的 地 名 标注 信 
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息 ,如 图 293. 1 的 左 图 所 示 。 单 击 “ 隐 和 藏 标注 信息 ”按钮 , 则 将 隐藏 在 百度 地 图 的 每 个 POI 的 地 名 标注 
信息 ,如 图 293. 1 的 右 图 所 示 。 


MySample 


显示 标注 信息 隐藏 标注 信息 显示 标注 信息 隐藏 标注 信息 


新 牌坊 立交 桥 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
BaiduMap myBaiduMap; 
MapView myMapView; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
SDKInitializer. initialize(getApplicationContext( ) ) ; 
setContentView(R. layout.activity main); 
myMapView = (MapView) findViewById(R. id.myMapView); 
myBaiduMap = myMapView. getMap( ); 
myMapView. showZoomControls(true); // 显 示 右 下 角 百 度 地 图 原生 的 放大 和 缩小 按钮 
// 重 庆 红 旗 河 沟 的 经 纬度 (106.53279,29.59129) 
LatLng myPoint = new LatLng(29.59129,106.53279); 
MapStatus myMapStatus = new MapStatus. Builder().target(myPoint).build(); 
myBaiduMap. setMapStatus(MapStatusUpdateFactory. newMapStatus(myMapStatus) ); 
MapStatusUpdate myMapStatusUpdate = MapStatusUpdateFactory. zoomTo(16 ) ; 
myBaiduMap. animateMapStatus(myMapStatusUpdate); 
} 
// 响 应 单 击 "显示 标注 信息 "按钮 
public void onClickBtnl (View v) { 
myBaiduMap. showMapPoi (true); 
} 
// 响 应 单 击 "隐藏 标注 信息 "按钮 
public void onClickBtn2(View v) { 
myBaiduMap. showMapPoi (false); 
} 
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上 面 这 段 代码 在 MyCode\ MySampleX77\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX77\ app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX77\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySampleX77\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms:play-services-appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 文 
件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/ sdk。 此 实例 的 完整 项 目 在 MyCode\MySampleX77 文件 夹 中 ， 


294 ”使 用 百度 SDK 实现 以 俯视 角 观 察 街 道 三 维 图 


此 实例 主要 通过 使 用 百度 SDK 的 MapStatus, 实 现在 指定 位 置 以 45 度 俯视 角 观 察 三 维 百 度 地 图 
街道 。 当 实例 运行 之 后 ,将 显示 以 重庆 江北 观音 桥 为 中 心 的 百度 地 图 ,如 图 294. 1 的 左 图 所 示 。 单 击 
“以 45 度 俯视 角 观 察 三 维 百 度 地 图 街道 ”按钮 , 则 该 地 图 上 的 建筑 物 将 呈现 三 维 效果 ,如 图 294. 1 的 


右 图 所 示 。 
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主要 代码 如 下 : 
public class MainActivity extends Activity { 
BaiduMap myBaiduMap; 
MapView myMapView; 
(@Override 


protected void onCreate( Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
SDKInitializer. initialize(getApplicationContext( ) ) ; 
setContentView(R. layout. activity main); 
myMapView = (MapView) findViewById(R. id.myMapView); 
myBaiduMap = myMapView. getMap( ); 
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myMapView. showZoomControls(true); 
// 重 庆 江 北 观音 桥 的 经 纬度 (106.539578, 29.579105) 
LatLng myPoint = new LatLng(29.579105,106.539578); 
MapStatus myMapStatus = new MapStatus. Builder().target(myPoint).build(); 
myBaiduMap. setMapStatus(MapStatusUpdateFactory. newMapStatus(myMapStatus) ) ; 
MapStatusUpdate myMapStatusUpdate = MapStatusUpdateFactory. zoomTo(18); 
myBaiduMap. animateMapStatus(myMapStatusUpdate); 

} 

// 响 应 单 击 " 以 45 度 俯视 角 观 察 三 维 百 度 地 图 街道 "按钮 

public void onClickBtnl (View v) { 
MapStatus myMapStatus = 

new MapStatus. Builder(). zoom(18).overlook( - 45).build( ); 
myBaiduMap. setMapStatus(MapStatusUpdateFactory. newMapStatus(myMapStatus) ); 
i 


上 面 这 上段 代码 在 MyCode\ MySampleX82\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 注 意 : 显示 百度 地 图 的 com. baidu. mapapi. map. MapView 控件 在 
MyCode\MySampleX82\app\src\main\res\layout\activity_main. xml 文件 中 添加 。 此 外 ,此 实例 需 
要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX82\app\libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\ 
MySampleX82\app\build. gradle 文件 中 添加 compile files('libs/BaiduLBS_ Android. jar') 和 compile 
'com. google. android. gms:play 一 services 一 appindexing: 8. 4.0' 依 赖 项 ,并 在 AndroidManifest. xml 
文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 
址 : http://lbsyun. baidu. com/sdk。 此 实例 的 完整 项 上 日 在 MyCode\MySampleX82 文件 夹 中 。 


295 ”使 用 百度 SDK 实现 根据 经 纬度 计算 两 地 距离 


此 实例 主要 通过 使 用 百度 地 图 SDK 的 DistanceUtil, 实 现 根据 经 纬度 值 计算 两 地 之 间 的 距离 。 
当 实 例 运 行 之 后 ,在 “ 甲 地 纬度 : ”输入 框 中 输入 重庆 的 纬度 值 ,如 “29. 615452”, 在 “ 甲 地 经 度 : ”输入 
框 中 输入 重庆 的 经 度 值 , 如 “106. 557317”; 在 “ 乙 地 纬度 : ”输入 框 中 输入 成 都 的 纬度 值 , 如 
“30.701717”, 在 “ 乙 地 经 度 : ”输入 框 中 输入 成 都 的 经 度 值 ,如 "104. 080302”, 然 后 单 击 按钮 “计算 甲乙 
两 地 的 直线 距离 ”, 则 在 弹出 的 Toast 中 显示 重庆 和 成 都 之 间 的 直线 距离 值 ,如 图 295. 1 的 左 图 所 示 。 
如 果 在 “ 甲 地 纬度 : ”输入 框 中 输入 重庆 的 纬度 值 ,如 “29. 615452”, 在 “ 甲 地 经 度 : ”输入 框 中 输入 重庆 
的 经 度 值 ,如 “106. 557317”; 在 “ 乙 地 纬度 : ”输入 框 中 输入 北京 的 纬度 值 ,如 “39. 900835”, 在 “ 乙 地 经 
度 : ”输入 框 中 输入 北京 的 经 度 值 ,如 “116. 328103”, 然 后 单 击 按 钮 “计算 甲乙 两 地 的 直线 距离 ”, 则 在 
弹出 的 Toast 中 显示 重庆 和 北京 之 间 的 直线 距离 值 ,如 图 295. 1 的 右 图 所 示 。 

主要 代码 如 下 : 


public void onClickBtn1 (View v) { // 响 应 单 击 " 计 算 甲 乙 两 地 的 直线 距离 "按钮 
LatLng myPointl1 = 
new LatLng(Double. valueOf (myEditPoint1A. getText().toString()), 
Double. valueOf (myEditPoint1B. getText().toString( ) ) ) ; 
LatLng myPoint2 = 
new LatLng(Double. valueOf (myEditPoint2A. getText( ) .toString() ) ， 
Double. valueOf (myEditPoint2B. getText().toString())); 
double myDistance = DistanceUtil. getDistance(myPoint1, myPoint2)/1000; 
Toast. makeText (MainActivity. this," 两 地 的 距离 是 :" 
+ myDistance + "公里 ", Toast.LENGTH SHORT). show( ) ; 
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中 重 
| 


MySample MySample 


甲 地 纬度 : jz9.615452 甲 地 纬度 : 9.615452 


甲 地 经 度 : 106.557317 甲 地 经 度 : 106.557317 


乙 地 纬度 : 30.701717 乙 地 纬度 : 39.900835 


乙 地 经 度 : 104.080302 乙 地 经 度 : 116.328103 


计算 甲乙 两 地 的 直线 距离 计算 甲乙 两 地 的 直线 距离 


两 地 的 距离 是 : 267.0185086451187 公 两 地 的 距离 是 : 1448.7480452449458 公 


也 


图 295.1 


上 面 这 段 代 码 在 MyCode\ MySampleX47\app\src\main\java\com\\bin\luo\mysample\ 
MainActivity. java 文件 中 。 此 外 ,此 实例 需要 引入 百度 地 图 组 件 , 即 MyCode\MySampleX47\ app\ 
libs 目录 下 的 所 有 内 容 , 然 后 在 MyCode\MySampleX47\app\ build. gradle 文件 中 添加 compile files 
('libs/BaiduLBS Android. jar') 和 compile 'com. google. android. gms: play-services-appindexing: 8. 4. 0 
依赖 项 ,并 在 AndroidManifest. xml 文件 中 添加 开发 密 钥 和 操作 权限 (参考 实例 279 或 直接 看 源 代 
人 码 )。 百 度 地 图 开发 说 明 请 参考 官方 网 址 : http://lbsyun. baidu. com/sdk。 此 实例 的 完整 项 目 在 
MyCode\MySampleX47 文件 夹 中 ， 


296 ”使 用 新 浪 SDK 实现 跳 转 到 微 博 主页 


此 实例 主要 通过 使 用 新 浪 微 博 SDK ,实现 从 当前 应 用 直接 跳 转 新 浪 微 博 的 个 人 账户 主页 。 当 实 
例 运行 之 后 , 单 击 按钮 “ 跳 转 到 新 沪 微 博 主 页 ”, 则 将 启动 新 浪 微 博 应 用 并 跳 转 到 个 人 账户 主页 ,效果 
分 别 如 图 296. 1 的 左 图 和 右 图 所 示 ，。 

主要 代码 如 下 : 


public class MainActivity extends Activity { 
RuthInfo myAuthInfo; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
myRuthInfo = new RuthInfo(this，"639620816" ，" http://www. sina.com", null); 


WbSdk. install(this, myAuthInfo); // 初 始 化 微 博 SDK 
setContentView(R. layout. activity main); 
} 
public void onClickBtnl (View v) { // 响 应 单 击 " 跳 转 到 新 浪 微 博 主页 "按钮 
WeiboPageUtils. getInstance(MainActivity. this, 
myAuthInfo). startUserMainPage("" ,false); // 在 微 博 应 用 中 显示 用 户 的 个 人 主页 


} } 
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分 


MySample 


跳 转 到 新 浪 微 博 主页 


全 部 微 请 (18) 


用 户 5635946622 


3 小 时 前 来 自 未 通过 审核 应 用 
啊 啊 啊 .2 网 页 链接 
L7 转发 


用 户 5635946622 


3 小 时 前 来 自 未 通过 审核 应 用 


啊 啊 啊 必 网 页 链接 


日 | 评论 


用 户 5635946622 


时 前 来 自 未 通过 审核 应 用 


图 296.1 


上 面 这 段 代 码 在 MyCode\ MySample935\app\src\main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 在 MyCode\MySample935\app\libs 目录 下 添 
加 库 文 件 core-4. 1. 0-openDefaultRelease. aar, 然 后 在 MyCode\MySample935\app 目录 下 的 build. 
gradle 文件 的 根 节 点 下 添加 如 下 代码 : 


repositories { flatDir { dirs 'libs'} } 


并 在 该 文件 的 dependencies 节点 下 添加 依赖 项 : 


compile 'com. sina. weibo. sdk:core:4.1.0:openDefaultRelease(@aar' 


同时 还 要 在 MyCode\MySample935\app\src\main\AndroidManifest. xml 文件 中 添加 下 列 权 限 : 


< uses - permission android:name = "android. permission. INTERNET" /> 
< uses - permission android: name = "android. permission. ACCESS WIFI STATE" /> 
<uses- permission android:name = "android. permission.ACCESS NETWORK STATE" /> 


此 外 ,使 用 新 浪 微 博 SDK 需要 在 微 博 开放 平台 上 创建 应 用 并 获取 AppKey, 应 用 创建 步骤 如 下 。 

(1) 登录 新 浪 微 博 开 放 平 台 (http://open. weibo. com/development/mobile), 单 击 “ 移 动 应 用 ”下 
方 的 “立即 接 入 ”按钮 ,并 在 弹出 的 对 话 框 中 单 击 “ 继 续 创 建 ” 按 钮 。 

(2) 在 弹出 的 页 面 表单 中 填 入 移动 应 用 相关 信息 ,完成 后 单 击 “ 创 建 ” 按 钮 即 可 完成 应 用 创建 
操作 。 

(3) 此 外 ,还 须 完 善 一 部 分 应 用 信息 , 即 需要 在 表单 内 填 人 应 用 包 名 、 应 用 签名 、 应 用 下 载 地 址 、 
Android 下 载 地 址 .应 用 人 简介、 应 用 介绍 等 必需 信息 。 

详细 情况 请 登录 新 浪 微 博 开 放 平 台 。 此 实例 的 完整 项 目 在 MyCode\MySample935 文件 夹 中 。 
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297 ”使 用 新 浪 SDK 获取 授权 微 博 账户 的 商 介 


此 实例 主要 通过 使 用 新 浪 微 博 SDK ,实现 在 登录 微 博 账户 成 功 之 后 ,从 当前 应 用 获取 新 沪 微 博 的 
个 人 账户 的 简介 信息 。 当 实例 运行 之 后 , 单 击 “ 登 录 微 博 账户 "按钮 , 则 执行 登录 微 博 账户 的 操作 ,如 
果 登 录 成 功 , 则 在 弹出 的 Toast 中 显示 “登录 成 功 !1”, 如 图 297. 1 的 左 图 所 示 。 只 有 在 登录 微 博 账户 
成 功 之 后 , 单 击 “获取 账户 信息 ”按钮 ,才能 在 弹出 的 Toast 中 显示 账户 的 简介 信息 ,如 图 297. 1 的 右 
图 所 示 。 


中 (2 = 


MySample MySample 


登录 微 博 账户 获取 账户 信息 登录 微 博 账户 获取 账户 信息 


用 户 昵 称 : 用 户 5635946622 
用 户 所 在 地 : 重庆 
用 户 性 别 : 男 
用 户 粉 丝 数 : 2 
用 户 关 注 数 : 154 
用 户 微 博 数 : 18 
用 户 收 藏 数 : 0 
登录 成 功 ! 


图 297.1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
AuthInfo myAuthInfo; 
SsoHandler myHandler; 
String myToken, myUid; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
myAuthInfo = new RuthInfo(this，"639620816" ，" http://www. sina.com", null); 
WbSdk. install(this, myAuthInfo); // 初 始 化 微 博 SDK 
setContentView(R. layout. activity_main); 
} 
// 啊 应 单 击 " 登 录 微 博 账 户 "按钮 
public void onClickBtnl (View v) { 
myHandler = new SsoHandler(this); 
myHandler. authorizeClientSso(new WbAuthListener() { 
(@ Override 
public void onSuccess(Oauth2RccessToken token) { 
myToken = token.getToken( ); 
myUid = token. getUid( ); // 获 取 授 权 Token 和 Uid 值 
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Toast. makeText (MainActivity.this, "登录 成 功 !", Toast. LENGTH_SHORT). show( ) ; 
} 
(Override 
public void cancel() { } 
(Override 
public void onFailure(WbConnectErrorMessage message) { } 
丰 
// 响 应 单 击 "获取 账户 信息 "按钮 
public void onClickBtn2(View v) { 
JsonObjectRequest myJsonRequest = new JsonObjectRequest( 
"https://api. weibo. com/2/users/show. json?access token=" + myToken + 
"Suid=" + myUid, null, new Response. Listener < JSONObject >() { 


(WOverride 
public void onResponse(JSONObject response) { // 请 求 当 前 授权 用 户 账 户 信息 
try { // 解 析 并 显示 用 户 信 息 


String myScreenName = response.getString(" screen _ name" ); 
String myLocation = response.getString(" location" ); 
String myFollowersCount = response.get("followers count" ).toString(); 
String myFriendsCount = response.get("friends count").toString( ) ; 
String myStatusesCount = response.get("statuses count").toString(); 
String myFavoritesCount = response.get("favourites count" ).toString(); 
String myGender = response.getString("gender").equals("m") ? 
" 男 ”: response. getString("gender" ).equals("f")?" 女 ”: "未 知 "; 
// 通 过 StringBuilder 拼接 字符 串 内 容 
StringBuilder myBuilder = new StringBuilder(); 
myBuilder.append(" 用户 昵称 :" + myScreenName + "\n"); 
myBuilder. append(" 用 户 所 在 地 :" + myLocation + "\n"); 
myBuilder.append(" 用户 性 别 :" + myGender + "\n"); 
myBuilder. append(" 用 户 粉 丝 数 :" + myFollowersCount + "\n"); 
myBuilder. append(" 用 户 关 注 数 :" + myFriendsCount + "\n"); 
myBuilder.append(" 用 户 微 博 数 :" + myStatusesCount + "\n"); 
myBuilder. append(" 用 户 收藏 数 :" + myFavoritesCount + "\n"); 
Toast. makeText(MainActivity. this, 
myBuilder. toString( ), Toast.LENGTH SHORT). show( ) ; 
} catch (Exception e) { e.printStackTrace( ); } 


} }, nu11); 
Volley. newRequestQueue(this).add(myJsonRequest); 
} 
(Override 
protected void onActivityResult(int requestCode, int resultCode, Intent data) { 
if (myHandler != null) // 通 过 SsoHandler 对 象 处 理 授权 回调 结果 
myHandler. authorizeCallBack(requestCode, resultCode, data); 
是 


上 F 面 这 段 代 码 在 MyCode\ MySample937\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 需 要 说 明 的 是 ,此 实例 需要 在 MyCode\MySample937\app\libs 目录 下 添 
加 库 文 件 core-4. 1. 0-openDefaultRelease. aar, 然 后 在 MyCode\MySample937\app 目录 下 的 build. 
gradle 文件 的 根 节 点 下 添加 如 下 代码 : 


repositories { flatDir { dirs 'libs'} } 


并 在 该 文件 的 dependencies 节点 下 添加 依赖 项 : 
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compile 'com. sina. weibo. sdk: core:4.1.0:openDefaultRelease(Waar' 
compile 'eu.the4thfloor. volley: com. android. volley:2015.05.28' 


同时 还 要 在 MyCode\MySample937\app\src\main\AndroidManifest. xml 文件 中 添加 下 列 权 限 : 


< uses - permission android: name = "android. permission. INTERNET" /> 
< uses - permission android:name = "android. permission. ACCESS WIFI STATE"/> 
< uses - permission android: name = "android. permission. ACCESS NETWORK STRTE" /> 


此 外 ,使 用 新 浪 微 博 SDK 服务 需要 在 微 博 开放 平台 上 创建 应 用 并 获取 AppKey ,详细 情况 请 登录 
新 浪 微 博 开 放 平 台 (http://open. weibo. com/ development/mobile)。 此 实例 的 完整 项 目 在 MyCode\ 
MySample937 文件 夹 中 ， 


298 ”使 用 新 沪 SDK 将 第 傅 账 户 人 简介 生成 二 维 码 


此 实例 主要 通过 使 用 新 浪 微 博 SDK ,实现 将 微 博 账户 的 简介 信息 生成 二 维 码 。 当 实例 运行 之 后 ， 
单 击 “将 微 博 账户 的 简介 信息 生成 二 维 码 ?按钮 , 则 执行 登录 微 博 账 户 的 操作 ,然后 将 登录 成 功 之 后 的 
微 博 账户 的 简介 信息 生成 二 维 码 ,如 图 298. 1 的 左 图 所 示 。 使 用 微 信 扫描 该 二 维 码 , 则 将 显示 该 微 博 
账户 的 简介 信息 ,如 图 298. 1 的 右 图 所 示 。 


中 国 草 "a Bs0% 18:2 ® ”dl B 90% 18:24 


MySample 


将 微 博 账户 的 简介 信息 生成 二 维 码 


图 298. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
RuthInfo myAuthInfo; 
SsoHandler myHandler; 
String myJsonParam = "{\"action name\":\"QR LIMIT SCENE\", 
\"action info\":" +"{\"scene\":{\"scene id\":123}}}"; 
ImageView myImageView; 
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Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState); 
myAuthInfo = new AuthInfo(this, "639620816", "http://www. sina.com", null); 
WbSdk. install(this, myAuthInfo); // 初 始 化 微 博 SDK 
setContentView(R. layout. activity_main); 
myImageView = (ImageView)findViewById(R. id.myImageView); 
} 
// 响 应 单 击 " 将 微 博 账户 的 简介 信息 生成 二 维 码 "按钮 
public void onClickBtnl (View v) { 
myHandler = new SsoHandler(this); 
// 设 置 授 权 登 录 回 调 监 听 器 
myHandler. authorizeClientSso(new WbAuthListener( ){ 
(@ Override 
// 登 录 成 功 时 的 回调 函数 
public void onSuccess(Oauth2RccessToken token){ 
try{ 
AccessTokenKeeper. writeAccessToken(MainActivity. this, token); 
JsonObjectRequest myTicketRequest = 
new JsonObjectRequest(Request. Method. POST ， 
"https://api. weibo. com/2/eps/qrcode/create?access token = ”+ 
token. getToken( ), new JSONObject (myJsonParam), 
new Response. Listener < JSONObject >( ){ 
@Override 
public void onResponse( JSONObject response){ 
try{ 
ImageRequest myRequest = new ImageRequest( 
"https: //api. weibo. com/2/eps/qrcode/ show?ticket =" + 
response. get("ticket" ).toString( ) ,new Response. Listener < Bitmap>(){ 
@Override 
public void onResponse(Bitmap bitmap){ 
myImageView. setImageBitmap(bitmap);)} 
},0,0,Bitmap. Config.RGB 565,null); 
Volley. newRequestQueue( MainActivity.this).add(myRequest ); 
} 
catch( JSONException e){e.printStackTrace( );} 
} },nu11); 
Volley. newRequestQueue(MainActivity. this).add(myTicketRequest); 
} 
catch(Exception e){e.printStackTrace( );} 
} 
(@ Override 
public void cancel( ){} 
(@ Override 
public void onFailure(WbConnectErrorMessage wbConnectErrorMessage){)} 
站 
(Override 
protected void onActivityResult(int requestCode, int resultCode, Intent data){ 
super. onActivityResult(requestCode, resultCode, data); 
// 通 过 SsoHandler 对 象 处 理 回 调 结 果 
if(myHandler!= null) myHandler. authorizeCallBack(requestCode, resultCode, data); 
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上 和 面 这 段 代 人 码 在 MyCode\ MySample938\app\src\main\java\com\bin\luo\ mysample\ 
MainActivity. java 文件 中 。 有 关 权 限 问 题 请 参考 实例 296 进行 修改 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample938 文件 夹 中 。 


299 ”使 用 新 浪 SDK 实现 搜索 指定 关键 字 的 微 博 
此 实例 主要 通过 使 用 新 浪 微 博 SDK ,实现 根据 搜索 关键 字 在 新 浪 微 博 中 搜索 指定 的 内 容 。 当 实 


例 运 行 之 后 ,在 “关键 字 : ”输入 框 中 输入 关键 字 , 如 “工商 约 谈 ”, 如 图 299. 1 的 左 图 所 示 ; 然后 单 击 
“搜索 微 博 ” 按 钮 , 则 该 关键 字 的 搜索 结果 将 出 现在 微 博 列表 中 ,如 图 299. 1 的 右 图 所 示 。 


MySample 


关键 字 : 红 商 约 谈 


-EE 


村 
@ 环球 时 报 宣 
刚刚 ”来 局 微 请 weit om 


【北京 工商 约 谈 " 样 音 " 要 求 即 时 阻 断 违规 直播 】 近 

日 ， 针 对 " 抖 音 " 短 视频 平台 涉嫌 发 布 售 假 视频 的 与 
情报 道 ， 北 京 市 工商 局 海淀 分 局 及 时 对 该 平台 经 营 
主体 北京 微 播 视界 科技 有 限 公司 进行 约 谈 。 约 谈 会 
上 ， 企业 负责 人 反馈 了 调查 情况 ， 表 示 针 对 平台 涉 
嫌 违 规 内 容 已 采取 删除 、 封 禁 措施 。 截 至 目前 ， 共 
查 删 视频 805 个 ... 全 文 


书 巴 空 欢 :什么 时 候 整 顿 拼 多 多 ? ? 名 名 
人 帅 鸡 儿 大 -: 把 拼 多 多 也 整顿 下 假 货 太 多 时 


图 299. 1 
主要 代码 如 下 : 


public class MainActivity extends Activity { 
AuthInfo myAuthInfo; 
EditText myEditText; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate( savedInstanceState) ; 
myRuthInfo = new RuthInfo(this,"639620816" ," http: //www. sina. com" ,null); 
WbSdk. install(this, myAuthInfo); // 初 始 化 微 博 SDK 
setContentView(R. layout. activity main); 
myEditText = (EditText) findViewById(R. id.myEditText); 
} 
// 啊 应 单 击 " 搜 索 微 博 " 按 钮 
public void onClickBtnl (View v) { 
// 获 取 微 博 工 具 类 实例 ,并 通过 该 实例 对 象 根据 指定 关键 字 搜 索 微 博 
WeiboPageUtils. getInstance( MainActivity. this,myAuthInfo). 
openWeiboSearchPage(myEditText. getText().toString(),false); 
} 
} 


i 0 


上 面 这 上段 代码 在 MyCode\ MySample930\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 有 关 权 限 问题 请 参考 实例 296 进行 修改 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample930 文件 夹 中 ， 


300 ”使 用 新 浪 SDK 实现 发 布 图 像 至 微 博 


此 实例 主要 通过 使 用 新 浪 微 博 SDK ,实现 将 图 像 发 布 到 新 浪 微 博 。 当 实例 运行 之 后 , 单 击 * 选 择 
图 像 ” 按 钮 ,然后 在 弹出 的 窗口 中 选择 一 幅 图 像 , 则 该 图 像 将 显示 在 下 面 的 ImageView 控件 中 ,如 
图 300. 1 的 左 图 所 示 。 单 击 “ 分 享 至 新 浪 微 博 ” 按 钮 , 则 在 ImageView 控件 中 显示 的 图 像 将 被 分 享 到 
新 浪 微 博 中 ,如 图 300. 1 的 右 图 所 示 。 


六 中 避 析 


转发 到 微 博 


MySample 


主要 代码 如 下 : 


public class MainActivity extends Activity { 
RuthInfo myAuthInfo; 
ImageView myImageView; 
WbShareHandler myHandler; 
(Override 
protected void onCreate(Bundle savedInstanceState) { 
super. onCreate(savedInstanceState) ; 
myRuthInfo = new RuthInfo(this，"639620816"”，"http://www. sina. com" ，nul1l); 


WbSdk. install(this, myAuthInfo); // 初 始 化 微 博 SDK 
myHandler = new WbShareHandler(this); 
myHandler. registerApp( ); // 注 册 应 用 


setContentView(R. layout. activity main); 
myImageView = (ImageView) findViewById(R. id.myImageView); 

} 

public void onClickBtnl (View v) { // 响 应 单 击 "选择 图 像 "按钮 
Intent myIntent = new Intent(Intent. ACTION PICK) ; 


> Android 炫 酷 应 用 300 例 ， 实战 篇 


myIntent. setType(" image/ * "); 
startActivityForResult(myIntent, 0); 


} 

(Override 

protected void onActivityResult( int requestCode, int resultCode, Intent data){ 
myImageView. setImageURI( data. getData( ) ) ; // 通 过 ImageView 控件 预览 所 选 图 像 

} 

public void onClickBtn2(View v) { // 响 应 单 击 " 分 享 至 新 浪 微 博 "按钮 


ImageObject myImageObject = new Image0bject(); 
myImageObject. setImageObject( ((BitmapDrawable) 
myImageView. getDrawable() ). getBitmap());  // 通 过 Image0bject 存储 图 像 内 容 
WeiboMultiMessage myMessage = new WeiboMultiMessage(); 
myMessage. imageObject = myImageObject; 
myHandler. shareMessage( myMessage, false); // 分 享 指定 图 像 至 新 浪 微 博 
局， 


上 面 这 段 代码 在 MyCode\ MySample933\app\src\ main\java\com\bin\luo\mysample\ 
MainActivity. java 文件 中 。 有 关 权 限 问 题 请 参考 实例 296 进行 修改 。 此 实例 的 完整 项 目 在 MyCode\ 
MySample933 文件 夹 中 。 


